当前位置: 首页 > news >正文

音视频入门基础:RTP专题(14)——FFmpeg源码中,对H.264的各种RTP有效载荷结构的解析

一、引言

由《音视频入门基础:RTP专题(10)——FFmpeg源码中,解析RTP header的实现》可以知道,FFmpeg源码的rtp_parse_packet_internal函数的前半部分实现了解析某个RTP packet的RTP header的功能。而在解析完RTP header后,rtp_parse_packet_internal函数内部会执行函数指针parse_packet指向的回调函数来对不同有效载荷类型的RTP payload进行解析:

static int rtp_parse_packet_internal(RTPDemuxContext *s, AVPacket *pkt,const uint8_t *buf, int len)
{
//...if (s->handler && s->handler->parse_packet) {rv = s->handler->parse_packet(s->ic, s->dynamic_protocol_context,s->st, pkt, &timestamp, buf, len, seq,flags);}
//...
}

比如,对于有效载荷类型为H.264的payload,parse_packet指向的回调函数为h264_handle_packet函数,此时通过h264_handle_packet函数对H.264格式的payload进行解析;对于有效载荷类型为AAC的payload,parse_packet指向的回调函数为aac_parse_packet函数,此时通过aac_parse_packet函数对AAC格式的payload进行解析。

二、h264_handle_packet函数的定义

h264_handle_packet函数定义在FFmpeg源码(本文演示用的FFmpeg源码版本为7.0.1)的源文件libavformat/rtpdec_h264.c:

// return 0 on packet, no more left, 1 on packet, 1 on partial packet
static int h264_handle_packet(AVFormatContext *ctx, PayloadContext *data,AVStream *st, AVPacket *pkt, uint32_t *timestamp,const uint8_t *buf, int len, uint16_t seq,int flags)
{uint8_t nal;uint8_t type;int result = 0;if (!len) {av_log(ctx, AV_LOG_ERROR, "Empty H.264 RTP packet\n");return AVERROR_INVALIDDATA;}nal  = buf[0];type = nal & 0x1f;/* Simplify the case (these are all the NAL types used internally by* the H.264 codec). */if (type >= 1 && type <= 23)type = 1;switch (type) {case 0:                    // undefined, but pass them throughcase 1:if ((result = av_new_packet(pkt, len + sizeof(start_sequence))) < 0)return result;memcpy(pkt->data, start_sequence, sizeof(start_sequence));memcpy(pkt->data + sizeof(start_sequence), buf, len);COUNT_NAL_TYPE(data, nal);break;case 24:                   // STAP-A (one packet, multiple nals)// consume the STAP-A NALbuf++;len--;result = ff_h264_handle_aggregated_packet(ctx, data, pkt, buf, len, 0,NAL_COUNTERS, NAL_MASK);break;case 25:                   // STAP-Bcase 26:                   // MTAP-16case 27:                   // MTAP-24case 29:                   // FU-Bavpriv_report_missing_feature(ctx, "RTP H.264 NAL unit type %d", type);result = AVERROR_PATCHWELCOME;break;case 28:                   // FU-A (fragmented nal)result = h264_handle_packet_fu_a(ctx, data, pkt, buf, len,NAL_COUNTERS, NAL_MASK);break;case 30:                   // undefinedcase 31:                   // undefineddefault:av_log(ctx, AV_LOG_ERROR, "Undefined type (%d)\n", type);result = AVERROR_INVALIDDATA;break;}pkt->stream_index = st->index;return result;
}

该函数的作用是:对H.264格式的RTP payload(有效载荷)进行解析。H.264格式的RTP的payload有三种不同的有效载荷结构:Single NAL Unit Packet、Aggregation Packet(STAP-A、STAP-B、MTAP16、MTAP24)和Fragmentation Unit(FU-A、FU-B)。在h264_handle_packet函数中对这些有效载荷结构进行统一解析处理。

形参ctx:输入型参数。用来输出日志,可忽略。

形参data:输入型参数,指向一个PayloadContext(有效载荷上下文)变量。

形参st:输入型参数,指向一个AVStream类型变量。该变量存贮该路视频流的信息。

形参pkt:输出型参数。执行h264_handle_packet函数后,pkt会得到从该RTP packet的payload中得到的信息。

形参timestamp:输入型参数,其指向的变量的值为该RTP packet的RTP header中的timestamp(时间戳)。

形参buf:输入型参数,指针buf指向该RTP packet的RTP payload的第一个字节,即RTP payload header。

形参len:输入型参数,为该RTP packet的RTP payload的大小(以字节为单位)。

形参seq:输入型参数,为该RTP packet的RTP header中的sequence number(序列号)。

形参flags:输入型参数,表示该RTP packet的RTP header中的marker字段的值是否为1。

返回值:返回一个负数表示失败,FFmpeg不支持这种有效载荷结构;返回非负数表示成功。

三、h264_handle_packet函数的内部实现分析

(一)解析RTP payload header 

h264_handle_packet函数内部,首先判断变量len(该RTP packet的RTP payload的大小)是否为0,如果为0,表示是空的H.264 RTP数据包,打印错误日志:“Empty H.264 RTP packet”:

    uint8_t nal;uint8_t type;int result = 0;if (!len) {av_log(ctx, AV_LOG_ERROR, "Empty H.264 RTP packet\n");return AVERROR_INVALIDDATA;}

由《音视频入门基础:RTP专题(12)——RTP封装H.264时,RTP中的NAL Unit Type》可以知道,如果RTP payload为H.264格式,那RTP payload的第一个字节就是RTP payload header,RTP payload header的结构就是NALU Header(包含forbidden_zero_bit、nal_ref_idc、nal_unit_type)。h264_handle_packet函数内部通过下面语句将RTP payload header的nal_unit_type读取出来,赋值给变量type:

    nal  = buf[0];type = nal & 0x1f;/* Simplify the case (these are all the NAL types used internally by* the H.264 codec). */if (type >= 1 && type <= 23)type = 1;

然后h264_handle_packet函数中会根据不同的nal_unit_type值执行不同的逻辑来处理不同的有效载荷结构。

(二)解析Single NAL Unit Packet

当nal_unit_type范围在1至 23(含 23)之间时,有效载荷结构为Single NAL Unit Packet,此时该RTP packet的有效载荷中只包含一个NALU,h264_handle_packet函数中会执行下面代码块来处理Single NAL Unit Packet:

    case 0:                    // undefined, but pass them throughcase 1:if ((result = av_new_packet(pkt, len + sizeof(start_sequence))) < 0)return result;memcpy(pkt->data, start_sequence, sizeof(start_sequence));memcpy(pkt->data + sizeof(start_sequence), buf, len);COUNT_NAL_TYPE(data, nal);break;

上面的代码块中,首先通过av_new_packet函数给pkt->data分配内存(关于av_new_packet函数用法可以参考:《FFmpeg源码:packet_alloc、av_new_packet、av_shrink_packet、av_grow_packet函数分析》),然后把该RTP packet的payload数据提取出来,加上值为“0001”(四字节)的起始码,存到pkt->data中:

数组start_sequence中存放的数据就是“0001”,表示H.264码流的NALU的起始码:

static const uint8_t start_sequence[] = { 0, 0, 0, 1 };

(三)解析STAP-A

当nal_unit_type值为24时,有效载荷结构为STAP-A,此时该RTP packet的有效载荷中可能包含多个NALU,h264_handle_packet函数中会执行下面代码块来处理STAP-A。

    case 24:                   // STAP-A (one packet, multiple nals)// consume the STAP-A NALbuf++;len--;result = ff_h264_handle_aggregated_packet(ctx, data, pkt, buf, len, 0,NAL_COUNTERS, NAL_MASK);break;

上面的代码块中,首先会执行语句:“buf++;len--;”让指针buf指向RTP payload header之后的数据,让len的值等于该RTP packet的RTP payload去掉RTP payload header之后的大小(以字节为单位)。然后会执行ff_h264_handle_aggregated_packet函数处理STAP-A。

ff_h264_handle_aggregated_packet函数定义在libavformat/rtpdec_h264.c中:

int ff_h264_handle_aggregated_packet(AVFormatContext *ctx, PayloadContext *data, AVPacket *pkt,const uint8_t *buf, int len,int skip_between, int *nal_counters,int nal_mask)
{int pass         = 0;int total_length = 0;uint8_t *dst     = NULL;int ret;// first we are going to figure out the total sizefor (pass = 0; pass < 2; pass++) {const uint8_t *src = buf;int src_len        = len;while (src_len > 2) {uint16_t nal_size = AV_RB16(src);// consume the length of the aggregatesrc     += 2;src_len -= 2;if (nal_size <= src_len) {if (pass == 0) {// countingtotal_length += sizeof(start_sequence) + nal_size;} else {// copyingmemcpy(dst, start_sequence, sizeof(start_sequence));dst += sizeof(start_sequence);memcpy(dst, src, nal_size);if (nal_counters)nal_counters[(*src) & nal_mask]++;dst += nal_size;}} else {av_log(ctx, AV_LOG_ERROR,"nal size exceeds length: %d %d\n", nal_size, src_len);return AVERROR_INVALIDDATA;}// eat what we handledsrc     += nal_size + skip_between;src_len -= nal_size + skip_between;}if (pass == 0) {/* now we know the total size of the packet (with the* start sequences added) */if ((ret = av_new_packet(pkt, total_length)) < 0)return ret;dst = pkt->data;}}return 0;
}

由《音视频入门基础:RTP专题(12)——RTP封装H.264时,RTP中的NAL Unit Type》可以知道,

当有效载荷结构(即RTP layload)为STAP-A时,此时:

该RTP数据包(RTP packet) = RTP header + STAP-A

一个STAP-A = RTP payload header(此时为STAP-A NAL HDR) + 若干个single-time aggregation units

一个single-time aggregation units = NAL unit size(固定占2字节) + NAL unit(包含NALU Header)

ff_h264_handle_aggregated_packet函数中,首先会通过AV_RB16宏定义将single-time aggregation unit的NAL unit size读取出来,存入变量nal_size中。关于AV_RB16宏定义的用法可以参考:《FFmpeg源码:AV_RB32、AV_RB16、AV_RB8宏定义分析》

 uint16_t nal_size = AV_RB16(src);

通过av_new_packet函数让指针dst(即pkt->data)指向一个分配的内存块:

        if (pass == 0) {/* now we know the total size of the packet (with the* start sequences added) */if ((ret = av_new_packet(pkt, total_length)) < 0)return ret;dst = pkt->data;}

如果NAL unit size超过剩下的RTP payload的大小,表示出错了,打印错误日志:"nal size exceeds length: %d %d\n"。如果没超过,根据NAL unit size的值把该RTP packet的payload中的每个NALU提取出来,每个NALU前都加上值为“0001”(四字节)的起始码,存到dst(即pkt->data)中:

            if (nal_size <= src_len) {if (pass == 0) {// countingtotal_length += sizeof(start_sequence) + nal_size;} else {// copyingmemcpy(dst, start_sequence, sizeof(start_sequence));dst += sizeof(start_sequence);memcpy(dst, src, nal_size);if (nal_counters)nal_counters[(*src) & nal_mask]++;dst += nal_size;}} else {av_log(ctx, AV_LOG_ERROR,"nal size exceeds length: %d %d\n", nal_size, src_len);return AVERROR_INVALIDDATA;}

执行ff_h264_handle_aggregated_packet函数后,pkt->data指向的缓冲区会得到该RTP packet的payload中的每个NALU的数据(可能包含多个NALU,每个NALU的数据之间以“0001”分隔)。

(四)解析FU-A

当nal_unit_type值为28时,有效载荷结构为FU-A,此时一个NALU可能会被分割成多个RTP  Packet,h264_handle_packet函数中会执行下面代码块来处理FU-A:

    case 28:                   // FU-A (fragmented nal)result = h264_handle_packet_fu_a(ctx, data, pkt, buf, len,NAL_COUNTERS, NAL_MASK);break;

h264_handle_packet_fu_a函数定义在libavformat/rtpdec_h264.c中:

static int h264_handle_packet_fu_a(AVFormatContext *ctx, PayloadContext *data, AVPacket *pkt,const uint8_t *buf, int len,int *nal_counters, int nal_mask)
{uint8_t fu_indicator, fu_header, start_bit, nal_type, nal;if (len < 3) {av_log(ctx, AV_LOG_ERROR, "Too short data for FU-A H.264 RTP packet\n");return AVERROR_INVALIDDATA;}fu_indicator = buf[0];fu_header    = buf[1];start_bit    = fu_header >> 7;nal_type     = fu_header & 0x1f;nal          = fu_indicator & 0xe0 | nal_type;// skip the fu_indicator and fu_headerbuf += 2;len -= 2;if (start_bit && nal_counters)nal_counters[nal_type & nal_mask]++;return ff_h264_handle_frag_packet(pkt, buf, len, start_bit, &nal, 1);
}

由《音视频入门基础:RTP专题(12)——RTP封装H.264时,RTP中的NAL Unit Type》可以知道,FU-A 由一个8位的碎片单元指示符(FU indicator,又称FU identifier,其实就是RTP payload header)、一个8位组的碎片单元报头(FU header)和一个碎片单元有效载荷(FU payload,又称fragmentation unit  payload,H264 NAL Unit Payload)组成。

h264_handle_packet_fu_a函数中,通过下面代码将FU indicator读取出来,存到变量fu_indicator中;将FU header读取出来,存到变量fu_header中;把fu_header的S位(起始位)读取出来,存到变量start_bit中;把fu_header的Type字段(表示NAL单元有效载荷类型)读取出来,存到变量nal_type中;变量nal相当于存贮该NALU的NALU Header:

    fu_indicator = buf[0];fu_header    = buf[1];start_bit    = fu_header >> 7;nal_type     = fu_header & 0x1f;nal          = fu_indicator & 0xe0 | nal_type;

让指针buf指向FU indicator和FU header之后的数据,即指向FU-A的FU payload。让变量len的值变为该FU payload的大小(以字节为单位):

    // skip the fu_indicator and fu_headerbuf += 2;len -= 2;

然后h264_handle_packet_fu_a函数中会调用ff_h264_handle_frag_packet函数处理该FU-A的FU payload:

    return ff_h264_handle_frag_packet(pkt, buf, len, start_bit, &nal, 1);

ff_h264_handle_frag_packet函数定义在libavformat/rtpdec_h264.c中。可以看到,执行ff_h264_handle_frag_packet函数后,pkt->data会得到该FU-A的FU payload(前面加上“0001”的起始码)中的数据,即得到该NALU在该RTP  Packet中的分片数据:

int ff_h264_handle_frag_packet(AVPacket *pkt, const uint8_t *buf, int len,int start_bit, const uint8_t *nal_header,int nal_header_len)
{int ret;int tot_len = len;int pos = 0;if (start_bit)tot_len += sizeof(start_sequence) + nal_header_len;if ((ret = av_new_packet(pkt, tot_len)) < 0)return ret;if (start_bit) {memcpy(pkt->data + pos, start_sequence, sizeof(start_sequence));pos += sizeof(start_sequence);memcpy(pkt->data + pos, nal_header, nal_header_len);pos += nal_header_len;}memcpy(pkt->data + pos, buf, len);return 0;
}

(五)解析其它有效载荷结构

当nal_unit_type值为25、26、27、29时,有效载荷结构分别为STAP-B、MTAP-16、MTAP-24、FU-B。由于FFmpeg目前(截止7.0.1版本)还不支持这几种有效载荷结构,所以h264_handle_packet函数中会通过avpriv_report_missing_feature函数打印错误日志:"RTP H.264 NAL unit type %d":

    case 25:                   // STAP-Bcase 26:                   // MTAP-16case 27:                   // MTAP-24case 29:                   // FU-Bavpriv_report_missing_feature(ctx, "RTP H.264 NAL unit type %d", type);result = AVERROR_PATCHWELCOME;break;

四、总结

1.FFmpeg源码中,在h264_handle_packet函数内部统一对H.264的各种RTP有效载荷结构进行解析处理。

2.FFmpeg目前(截至7.0.1版本)还不支持STAP-B、MTAP-16、MTAP-24、FU-B这几种有效载荷结构的解析。所以如果要解析包含这几种有效载荷结构的RTP流,可能会出错。要想支持,可以修改FFmpeg源码,在h264_handle_packet函数内部添加解析对应的有效载荷结构的代码。

相关文章:

音视频入门基础:RTP专题(14)——FFmpeg源码中,对H.264的各种RTP有效载荷结构的解析

一、引言 由《音视频入门基础&#xff1a;RTP专题&#xff08;10&#xff09;——FFmpeg源码中&#xff0c;解析RTP header的实现》可以知道&#xff0c;FFmpeg源码的rtp_parse_packet_internal函数的前半部分实现了解析某个RTP packet的RTP header的功能。而在解析完RTP head…...

FPGA——4位全加器及3-8译码器的实现

文章目录 一、全加器1、Verilog实现四位全加器2、下载测试 二、3-8译码器1、Verilog实现3-8译码器2、7段数码管显示3-8译码器 三、总结四、参考资料 一、全加器 全加器的定义&#xff1a; 全加器英语名称为full-adder&#xff0c;是用门电路实现两个二进制数相加并求出和的组合…...

软考中级-数据库-3.4 数据结构-图

图的定义 一个图G(Graph)是由两个集合:V和E所组成的&#xff0c;V是有限的非空顶点(Vertex)集合&#xff0c;E是用顶点表示的边(Edge)集合&#xff0c;图G的顶点集和边集分别记为V(G)和E(G)&#xff0c;而将图G记作G(V&#xff0c;E)。可以看出&#xff0c;一个顶点集合与连接这…...

软考中级-数据库-3.3 数据结构-树

定义:树是n(n>=0)个结点的有限集合。当n=0时称为空树。在任一非空树中,有且仅有一个称为根的结点:其余结点可分为m(m>=0)个互不相交的有限集T1,T2,T3...,Tm…,其中每个集合又都是一棵树,并且称为根结点的子树。 树的相关概念 1、双亲、孩子和兄弟: 2、结点的度:一个结…...

Win11被背刺,官方泄露免费激活方法

AI已经成为科技圈的主旋律了&#xff0c;在PC圈的龙头微软也不例外。 但最近喜欢背刺用户、极力推崇AI的微软被自家产品背刺了一把。 罪魁祸首就是Microsoft Copilot&#xff0c;如果向Microsoft Copilot提问&#xff0c;是否可以帮忙提供激活Windows11的脚本。 Copilot会立马…...

第十天-字符串:编程世界的文本基石

在编程的广阔领域中&#xff0c;字符串是极为重要的数据类型&#xff0c;它就像一座桥梁&#xff0c;连接着人类的自然语言和计算机能够理解与处理的数字信息。下面&#xff0c;让我们深入探索字符串的世界。 一、字符串简介 字符串是由零个或多个字符组成的有序序列&#xff…...

CentOS7 安装Redis 6.2.6 详细教程

本文主要介绍CentOS7系统下安装Redis6.2.6的详细教程。 1.安装依赖 redis是基于C语言开发&#xff0c;因此想要在服务器上运行redis需要验证是否安装了gcc&#xff0c;没有安装gcc则需先安装 查看是否安装gcc gcc -v如果没有安装gcc&#xff0c;则通过如下命令安装 yum in…...

VsCode使用

vscode前端vue项目启动&#xff1a;Vue项目的创建启动及注意事项-CSDN博客 vscode使用教程&#xff1a;史上最全vscode配置使用教程 - 夏天的思考 - 博客园 vscode如何从git拉取代码&#xff1a;vscode如何从git拉取代码 • Worktile社区...

mac上最好的Python开发环境之Anaconda+Pycharm

文章目录 一、前言 1. Anaconda介绍2. Pycharm介绍 编码协助项目代码导航代码分析Python重构支持Django框架集成版本控制 二、下载Anaconda和Pycharm 1. 下载Anaconda2. 下载Pycharm 三、安装Anaconda和Pycharm 1. 安装Anaconda2. 安装Pycharm 一、前言 1. Anaconda介绍 …...

防火墙旁挂组网双机热备负载均衡

一&#xff0c;二层交换网络&#xff1a; 使用MSTPVRRP组网形式 VLAN 2--->SW3为主,SW4 作为备份 VLAN 3--->SW4为主,SW3 作为备份 MSTP 设计 --->SW3 、 4 、 5 运行 实例 1 &#xff1a; VLAN 2 实例 2 &#xff1a; VLAN 3 SW3 是实例 1 的主根&#xff0c;实…...

Docker 学习(三)——数据管理

容器中的管理数据主要有两种方式&#xff1a; 数据卷 &#xff08;Data Volumes&#xff09;&#xff1a; 容器内数据直接映射到本地主机环境&#xff1b; 数据 卷容器&#xff08; Data Volume Containers&#xff09;&#xff1a; 使用特定容器维护数据卷 1.数据卷 数据卷…...

中间件专栏之MySQL篇——MySQL缓存策略

本文所说的MySQL缓存策略与前文提到的buffer pool不同&#xff0c;那是MySQL内部自己实现的&#xff0c;本问所讲的缓存策略是使用另一个中间件redis来缓存MySQL中的热点数据。 一、为什么需要MySQL缓存方案 缓存用户定义的热点数据&#xff0c;用户可以直接从缓存中获取热点…...

高频 SQL 50 题(基础版)_196. 删除重复的电子邮箱

高频 SQL 50 题&#xff08;基础版&#xff09;_196. 删除重复的电子邮箱 思路 思路 DELETE p1 FROM Person p1,Person p2 WHEREp1.Email p2.Email AND p1.Id > p2.Id...

github进不去,一直显示错误

1、进入网址Dns检测|Dns查询 - 站长工具 2、复制检测出来的任意一个ip 3、打开电脑的文件夹&#xff1a;C:\Windows\System32\drivers\etc 下的hosts文件下复制这个ip地址 20.205.243.166 4、winr 打开cmd&#xff0c;输入ipconfig/flushdns ipconfig/flushdns出现这个就可以…...

MWC 2025|美格智能发布基于高通®X85 5G调制解调器及射频的新一代5G-A通信模组SRM819W

3月3日&#xff0c;在MWC 2025世界移动通信大会上&#xff0c;美格智能正式推出基于高通X85调制解调器及射频的新一代5G-A通信模组SRM819W&#xff0c;集5G-A、毫米波、AI加持的网络优化等最前沿的通信技术&#xff0c;成为行业首批搭载高通X85的5G通信模组产品&#xff0c;将助…...

【零基础到精通Java合集】第十集:List集合框架

课程标题:List集合框架(15分钟) 目标:掌握List接口核心实现类(ArrayList/LinkedList)的使用与场景选择,熟练操作有序集合 0-1分钟:List概念引入 以“购物清单”类比List特性:元素有序(添加顺序)、可重复、支持索引访问。说明List是Java集合框架中最常用的数据结构…...

《今日-AI-编程-人工智能日报》

一、AI行业动态 荣耀发布“荣耀阿尔法战略” 荣耀在“2025世界移动通信大会”上宣布&#xff0c;将从智能手机制造商转型为全球领先的AI终端生态公司&#xff0c;并计划未来五年投入100亿美元建设AI设备生态。荣耀展示了基于GUI的个人移动AI智能体&#xff0c;并推出多款AI终端…...

在 MyBatis 中,若数据库字段名与 SQL 保留字冲突解决办法

在 MyBatis 中&#xff0c;若数据库字段名与 SQL 保留字冲突&#xff0c;可通过以下方法解决&#xff1a; 目录 一、使用转义符号包裹字段名二、通过别名映射三、借助 MyBatis-Plus 注解四、全局配置策略&#xff08;辅助方案&#xff09;最佳实践与注意事项 一、使用转义符号…...

从基础到实践(十):MOS管的全面解析与实际应用

MOS管&#xff08;金属-氧化物半导体场效应晶体管&#xff09;是现代电子技术的基石&#xff0c;凭借高输入阻抗、低功耗和易集成特性&#xff0c;成为数字电路、电源管理和信号处理的核心元件。从微处理器到新能源汽车电驱系统&#xff0c;其高效开关与放大功能支撑了计算机、…...

电源测试系统有哪些可以利用AI工具的科技??

AI技术的发展对电源模块测试系统的影响是深远的&#xff0c;不仅协助系统提升了测试效率和精度&#xff0c;还推动了测试方法的创新和智能化。那么在电源测试系统中哪些模块可以利用AI工具实现自动化测试? 1. 自动化测试与效率提升 智能测试流程优化 AI算法可以自动优化测试…...

RabbitMQ 最新版:安装、配置 与Java 接入详细教程

目录 一、RabbitMQ 简介二、RabbitMQ 的安装1. 安装 Erlang下载 Erlang安装 Erlang2. 安装 RabbitMQ下载 RabbitMQ安装 RabbitMQ3. 配置环境变量4. 启用管理插件三、RabbitMQ 的配置1. 创建用户和设置权限2. 配置文件四、Java 接入 RabbitMQ1. 添加依赖2. 创建连接3. 创建通道4…...

股市现期驱动因子

在股票投资中&#xff0c;我们把驱动股市收益的基本元素称为基本因素&#xff1a; 例如资产负债、现金流量 从短期来看&#xff0c;股市的上涨和下跌基于市场情绪&#xff0c;它更依赖于投资者的期望&#xff0c;投它涨的人多&#xff0c;它就涨。从长期来看&#xff0c;股市…...

物联网中的气象监测设备具备顶级功能

物联网中的气象监测设备具备顶级功能时&#xff0c;通常集成GPS、数据上报和预警系统&#xff0c;以确保精准监测和及时响应。以下是这些功能的详细说明&#xff1a; 1. GPS定位 精准定位&#xff1a;GPS模块提供设备的精确地理位置&#xff0c;确保数据与具体位置关联&#…...

算法1-4 凌乱的yyy / 线段覆盖

题目描述 现在各大 oj 上有 n 个比赛&#xff0c;每个比赛的开始、结束的时间点是知道的。 yyy 认为&#xff0c;参加越多的比赛&#xff0c;noip 就能考的越好&#xff08;假的&#xff09;。 所以&#xff0c;他想知道他最多能参加几个比赛。 由于 yyy 是蒟蒻&#xff0c…...

gn学习存档

以下答案均由deepseek提供&#xff0c;仅作学习存档。 1. 举例说明action和action_foreach区别 场景设定 假设需要处理一组文件&#xff1a; 输入文件&#xff1a;src/data/file1.txt, src/data/file2.txt, src/data/file3.txt处理逻辑&#xff1a;将每个 .txt 文件转换为 …...

SQL注入练习场:PHPStudy+SQLI-LABS靶场搭建教程(零基础友好版)

注意&#xff1a;文中涉及演示均为模拟测试&#xff0c;切勿用于真实环境&#xff0c;任何未授权测试都是违法行为&#xff01; 一、环境准备 下载PHPStudy 官网下载地址&#xff1a;https://www.xp.cn/php-study&#xff08;选择Windows版&#xff09; 安装时建议选择自定…...

python学习笔记——Thread常用方法

Thread对象中的一些方法&#xff1a; 以前说过多线程&#xff0c;用到threading模块中的Thread对象&#xff0c;其中的start和run方法比较熟悉了&#xff0c;start&#xff08;&#xff09;是重载了Thread对象中的run方法&#xff0c;其实作用还是&#xff0c;当执行这个start…...

2024年数学SCI2区TOP:雪雁算法SGA,深度解析+性能实测

目录 1.摘要2.算法原理3.结果展示4.参考文献5.代码获取 1.摘要 本文提出了一种雪雁算法&#xff08;SGA&#xff09;&#xff0c;该算法借鉴了雪鹅的迁徙行为&#xff0c;并模拟了其迁徙过程中常见的“人字形”和“直线”飞行模式。 2.算法原理 雪雁以其卓越的长途迁徙能力和…...

Kubernetes 指令备忘清单

文章目录 查看资源信息节点容器组命名空间无状态服务守护进程集事件服务帐户日志副本集角色保密字典配置项路由持久卷持久卷声明存储类多个资源 变更资源属性污点标签维护/可调度清空节点节点/容器组无状态/命名空间服务守护进程集服务账号注释 添加资源创建容器组创建服务创建…...

servlet tomcat

在spring-mvc demo程序运行到DispatcherServlet的mvc处理 一文中&#xff0c;我们实践了浏览器输入一个请求&#xff0c;然后到SpringMvc的DispatcherServlet处理的整个流程. 设计上这些都是tomcat servlet的处理 那么究竟这是怎么到DispatcherServlet处理的&#xff0c;本文将…...

在 Ubuntu 系统 22.04 上安装 Docker

在 Ubuntu 系统 22.04 上安装 Docker 在 Ubuntu 系统 22.04 上安装 Docker1. 更新系统包2. 安装依赖工具3. 添加 Docker 官方 GPG 密钥4. 添加 Docker 的 APT 仓库5. 安装 Docker Engine6. 启动并设置 Docker 服务7. 验证安装8. 配置非 Root 用户权限&#xff08;可选&#xf…...

一分钟理解Mybatis 里面的缓存机制

​ MyBatis 是一个流行的 Java 持久层框架&#xff0c;它简化了数据库操作。MyBatis 提供了强大的缓存机制&#xff0c;用于提升性能&#xff0c;减少数据库的访问次数。MyBatis 的缓存机制分为一级缓存和二级缓存。 ​ 该图展示了用户通过 SqlSession 发起查询请求&#xff0c…...

【我的Android进阶之旅】如何使用NanoHttpd在Android端快速部署一个HTTP服务器?

文章目录 开篇:程序员的"摸鱼神器"?一、为什么选择NanoHttpd?二、五分钟极速上车指南2.1 ▶ 第一步:引入依赖的哲学2.2 ▶ 第二步:创建服务器类:继承大法好2.3 ▶ 第三步:启动服务的仪式感三、高级玩法:让服务器不再单调3.1 🔥 场景1:变身文件服务器3.2 �…...

PyCharm 无法识别 Conda 环境的解决方案

一、问题分析 当在最新版 PyCharm (2024.3) 中配置 Conda 环境时&#xff0c;可能会出现以下典型错误&#xff1a; 找不到 Conda 可执行文件 我在网上找了很多解决办法&#xff0c;都没有有效解决这个问题&#xff0c;包括将环境路径替换为 .bat 文件和查找 python.exe 文件…...

AutoGen学习笔记系列(一)Tutorial - Model

这个系列文章记录了学习微软 AutoGen 的过程&#xff0c;与 smolagents 学习笔记系列一样&#xff0c;仍然以官方教程自己的理解为主线&#xff0c;中间可能穿插几个番外支线的形式写博客。 【注意】&#xff1a;在阅读这篇文章之前需要确保已经按照其 Installation 小节完成必…...

利用Git和wget批量下载网页数据

一、Git的下载&#xff08;参考文章&#xff09; 二. wget下载&#xff08;网上很多链接&#xff09; 三、git和wget结合使用 1.先建立一个文本&#xff0c;将代码写入文本&#xff08;代码如下&#xff09;&#xff0c;将txt后缀改为sh&#xff08;download_ssebop.sh&#xf…...

多线程JUC(一)

目录 前言一、多线程的三种实现方式1.继承Thread类2.实现Runnable接口3.利用Callable接口和Future接口4.三种方式对比 二、常见的成员方法1.getName、setName、currentThread、sleep2.线程的优先级3.守护线程4.插入线程 三、线程安全1.线程的生命周期2.同步代码块3.同步方法4.l…...

夸父工具箱(安卓版) 手机超强工具箱

如今&#xff0c;人们的互联网活动日益频繁&#xff0c;导致手机内存即便频繁清理&#xff0c;也会莫名其妙地迅速填满&#xff0c;许多无用的垃圾信息悄然占据空间。那么&#xff0c;如何有效应对这一难题呢&#xff1f;答案就是今天新推出的这款工具软件&#xff0c;它能从根…...

2025系统架构师(一考就过):案例之五:典型架构、架构演化、人工智能、云计算、大数据

六、中间件技术、典型架构 ◆中间件:在一个分布式系统环境中处于操作系统和应用程序之间的软件&#xff0c;可以在不同的技术之间共享资源&#xff0c;将不同的操作系统、数据库、异构的网络环境以及若干应用结合成一个有机的协同工作整体。 ◆中间件位于客户机/服务器的操作系…...

【随手笔记】利尔达NB模组

1.名称 移芯EC6263GPP 参数 指令备注 利尔达上电输出 [2025-03-04 10:24:21.379] I_AT_WAIT:i_len2 [2025-03-04 10:24:21.724] LI_AT_WAIT:i_len16 [2025-03-04 10:24:21.724] [2025-03-04 10:24:21.733] Lierda [2025-03-04 10:24:21.733] [2025-03-04 10:24:21.745] OK移…...

Mybatis 中#{} 和${} 的区别是什么?

在 MyBatis 中&#xff0c;#{} 和 ${} 都是用于动态 SQL 语句中的占位符&#xff0c;但是它们的作用和使用方式是不同的。下面是它们的区别&#xff1a; 1. #{} —— 用于防止 SQL 注入和自动类型处理 #{} 是用来将参数安全地传递到 SQL 语句中&#xff0c;它会将传递的参数值…...

nginx+keepalived负载均衡及高可用

1 项目背景 keepalived除了能够管理LVS软件外&#xff0c;还可以作为其他服务的高可用解决方案软件。采用nginxkeepalived&#xff0c;它是一个高性能的服务器高可用或者热备解决方案&#xff0c;Keepalived主要来防止服务器单点故障的发生问题&#xff0c;可以通过其与Nginx的…...

数据结构理论

目录 基本概念和术语 数据 数据元素 数据项 数据对象 数据结构 数据的结构 逻辑结构 存储结构&#xff08;物理结构&#xff09; 数据类型 定义 原子数据类型 结构数据类型 抽象数据类型&#xff08;Abstract Data Type&#xff0c;ADT&#xff09; 算法和算法分…...

【心得】一文梳理高频面试题 HTTP 1.0/HTTP 1.1/HTTP 2.0/HTTP 3.0的区别并附加记忆方法

面试时很容易遇到的一个问题—— HTTP 1.0/HTTP 1.1/HTTP 2.0/HTTP 3.0的区别&#xff0c;其实这四个版本的发展实际上是一环扣一环的&#xff0c;是逐步完善的&#xff0c;本文希望帮助读者梳理清楚各个版本之间的区别&#xff0c;并且给出当前各个版本的应用情况&#xff0c;…...

在Spring Boot项目中导出复杂对象到Excel文件

在Spring Boot项目中导出复杂对象到Excel文件&#xff0c;可以利用Hutool或EasyExcel等库来简化操作。这里我们将详细介绍如何使用Hutool和EasyExcel两种方式来实现这一功能。 使用Hutool导出复杂对象到Excel 首先确保你的pom.xml中添加了Hutool的依赖&#xff1a; <depe…...

spark 常见操作命令

配置虚拟机 配置即让自己的虚拟机可以联网&#xff0c;和别的虚拟机通讯 一、配置vm虚拟机网段。 具体设置为&#xff1a;虚拟机左上角点击编辑→虚拟网络编辑器&#xfffc; 选择VMnet8&#xff0c; 要改动两个地方&#xff08;注意&#xff1a;它会需要管理员权限&#xff…...

深入理解设计模式中的工厂模式(Factory Pattern)

各类资料学习下载合集 ​​https://pan.quark.cn/s/8c91ccb5a474​​ 工厂模式是创建对象的一种设计模式,属于创建型设计模式。它提供了一种方法来创建对象,而无需在代码中直接指定对象的具体类。工厂模式通过将对象的创建过程封装起来,使得代码更加灵活、可维护…...

DPDK网络开发

DPDK&#xff08;Data Plane Development Kit&#xff09;是一个用于快速数据包处理的开源库&#xff0c;广泛应用于高性能网络应用开发。 环境准备 硬件要求 NIC&#xff08;网络接口卡&#xff09;&#xff1a;支持DPDK的网卡&#xff0c;如Intel的82599、X710等。 CPU&am…...

第三节:基于Winform框架的串口助手小项目---串口操作《C#编程》

知识是无尽的宝藏&#xff0c;学习的过程虽有挑战&#xff0c;但每一次突破都是对自我的升华&#xff0c;向着更优秀的自己全力进发。 -----------WHAPPY 本节将重点介绍&#xff0c;如何修改控件的属性、SerialPort类的使用及实现串口初始化的操作 1.修改控件属性 修改属性…...

机器学习核函数

在机器学习中&#xff0c;核函数&#xff08;Kernel Function&#xff09;是一个非常重要的概念&#xff0c;特别是在支持向量机&#xff08;SVM&#xff09;等算法中有着广泛的应用。下面从定义、作用、常见的核函数类型、工作原理等方面详细介绍&#xff1a; 定义&#xff1…...