- 浏览: 525992 次
- 性别:
- 来自: 杭州
文章分类
- 全部博客 (161)
- 多线程与并发编程 (20)
- 算法和数据结构 (8)
- 缓存 (0)
- HttpClient (2)
- 疑难杂症 (11)
- Java内存管理 (17)
- 分布式开发 (14)
- Linux常用命令 (10)
- OSGI (0)
- UML (2)
- 趣味面试题 (5)
- 设计模式 (8)
- Java类加载 (2)
- JSTL (1)
- Web 服务器 (4)
- IT人生 (3)
- Java基础 (11)
- Flash技术 (7)
- 新知识 (3)
- 常用速备速查 (4)
- 版本控制 (1)
- Java集合工具类 (6)
- web前端技术 (1)
- 趣味话题 (1)
- 安全 (0)
- 架构设计 (5)
- Spring (4)
- 负载均衡技术 (2)
- 持久层技术 (2)
- MySQL单机多实例方案 (1)
- 收藏备用 (0)
- 性能优化 (3)
最新评论
-
liuwuhen:
...
Pushlet的工作原理 -
fbwfbi:
fengchuizhuming 写道楼主的完全正确。鉴定完毕楼 ...
硬件同步原语(CAS)理论 -
passerby_whu:
uule 写道这个测试后结果为:“testPageConten ...
FutureTask的使用方法和使用实例 -
fengchuizhuming:
楼主的完全正确。鉴定完毕
硬件同步原语(CAS)理论 -
edwardjuice:
FutureTask的使用方法和使用实例
有新的项目,关于RTMP的,挺新鲜的东西,这篇文章比较全面介绍了RTMP协议的内容,作为参考资料记录之。
RTMP 协议研究
1 协议研究概述
协议设计和分析一直都是在工作遇到,正好在这里总结一下,说到协议,在这个网络的时代,没有人可以离开它了。他存在我们生活中的任何角落,只不过我们平
时,并没有注意到它的存在,可以这么说如果没有协议,我们生活和日常的工作生产都不能进行。如果仔细想想你生活中用到的所有东西,协议已经包含其中。那到
底什么是协议呢?说的简单一点就是双方达成的共识,以便更好的交流,理论上协议是什么呢?如果学过《信号与系统》的人都知道有个简单的道理,就是信息在经
过一个管道的符号集,到另一个符号集时信息不会丢失。
任何复杂的事物都有个最简单的本质,网络上的协议也是这样,有个最基本的本质。除去上下层的概念,协议就只剩下通信双方实体的规则。
一般的协议都包含最基本的协议头,不管是物理层、链路层、还是网络层,这个头就构成了协议的本质东西。通常协议头要包含以下最基本的三项信息:
双方实体的唯一标示,用来标示通信双方的实体。
类型描述或者是净核描述,标志净核的内容。
协议净核的长度,用来在萃取净核的内
容应用。
其中,前两项是必须要有的,没有他们,通信双方的交互根本得不到保证,第三项在不太灵活的通信中可以去掉,而有第二项的类型推出。
协议的丰富性,有净核的多样性体现。
协议头除了以上的三项,还可以增加更多的信息(比如控制信息、时间信息等),取决于具体的应用。找到这些基本的东西,再去看协议的时候,能够更好的抓住协 议的主体进行分析和设计了。
2 RTMP 协议概述
RTMP 协议是被 Flash
用于对象、视频、音频的传输。该协议建立在 TCP 协议或者轮询 HTTP 协议之上, RTMP 协议就像一个用来装数据包的容器,这些数据可以是
AMF 格式的数据,也可以是 FLV 中的视 /
音频数据。一个单一的连接可以通过不同的通道传输多路网络流,这些通道中的包都是按照固定大小的包传输的 .
3 RTMP 协议部分
3.1 协议头
struct RTMP_HEAD
{
char cChannelid : 6;// 第一个字节的后 6 位
char cCheadsize ; // 第一个字节的头两位
char cTimer[3]; // 三个字节表示的时间信息
char cLength[3]; // 三个字节表示的长度
char cDatatype; // 数据类型
char sStreamid[4]; // 流标识
};
这里有三个最基本的元素(唯一标示 )、(类型 )和(净核的长度 )分别是: cChannelid 、 cDatatype 和 cLength 。
3.2 数据类型
数据类型 决定了协议上层可以做的具体的事情,和使用协议的人必须遵循的规则。
同时数据类型 说明了净核 的基本内容。
RTMP 数据类型:
0×01, Chunk Size, changes the chunk size for packets
0×02,
Unknown, anyone know this one?
0×03, Bytes Read, send
every x bytes read by both sides
0×04, Ping, ping is a
stream control message, has subtypes
0×05, Server BW, the
servers downstream bw
0×06, Client BW, the clients upstream
bw
0×07, Unknown, anyone know this one?
0×08, Audio
Data, packet containing audio
0×09, Video Data, packet
containing video data
0×0A-0×11, Unknown, anyone know?
0×12,
Notify, an invoke which does not expect a reply
0×13,
Shared Object, has subtypes
0×14, Invoke, like remoting
call, used for stream actions too.
3.3 协议的净核
RTMP 的协议净核是用 AMF 格式来描述, AMF
格式本身的产生就是为了 RTMP 协议服务的,最初的 RTMP 采用 XML 的形式传输数据,但 XML
只是字符形式的值对的格式传输数据,而随着应用的普及这完全不能满足要求了,比如对象、结构、数组,甚至可以是数据集,配合 DataGrid
组件可以很方便地显示数据。
为了处理复杂数据类型,采用一种独有的方式使 Flash 与应用服务器间可以来回传送数据势在必行。于是 AMF
应运而生。
AMF 是 Adobe 独家开发出来的通信协议,它采用二进制压缩,序列化、反序列化、传输数据,从而为 Flash 播放器与 Flash
Remoting 网关通信提供了一种轻量级的、高效能的通信方式。
AMF 最大的特色在于可直接将 Flash 内置对象,例如
Object, Array, Date, XML
,传回服务器端,并且在服务器端自动进行解析成适当的对象,这就减轻了开发人员繁复工作,同时也更省了开发时间。由于 AMF
采用二进制编码,这种方式可以高度压缩数据,因此非常适合用 来传递大量的资料。数据量越大, Flash Remoting
的传输效能就越高,远远超过 Web Service 。至于 XML, LoadVars 和 loadVariables()
,它们使用纯文本的传输方式,效能就更不能与 Flash Remoting 相提并论了。
注意: Flash Remoting 需要浏览器支持 Binary POST , Flash 播放器在 Netscape
6.x. 环境下运行 Flash Remoting 会不起作用( Flash Remoting 调用没有效果也不返回错误), Netscape 7
已经纠正了这个 bug 。对于早期 Safari 和 Chimera 版的苹果机也有这个问题。
同样是轻量级数据交换协议,同样是通过调
用远程服务,同样是基于标准的 HTTP 和 HTTPS 协议, Flash Remoting 为什么选择了使用 AMF 而放弃了 SOAP 与
Flash 播放器通信呢 有如下原因:
SOAP 将数据处理成 XML 格式,相对于二进制的 AFM 太冗长了;
AMF
能更有效序列化数据;因为 AMF 的初衷只是为了支持 Flash ActionScript 的数据类型,而 SOAP
却致力于提供更广泛的用途;
AMF 支持 Flash 播放器 6 只需要浏览器增加 4 KB 左右(压缩后)的大小,而 SOAP
就大多了;
SOAP 的一些头部文件请求在 Flash 播放器 6 不支持。那 Flash 播放器 6 为什么能访问基于 SOAP 的 Web 服务呢?原来 Flash Remoting 网关将 SOAP 请求在服务器端与转换成 AFM 格式,然后利用 AFM 与 Flash 播放器通信。
另外, AMF 包中包含 onResult 事件(比如说 response 事件)和 onStatus 事件(比如说 error
事件),这些事件对象在 Flash 中可以直接使用。
AMF 从 Flash MX 时代的 AMF0 发展到现在的 AMF3 。
AMF3 用作 Flash Playe 9 的 ActionScript 3.0 的默认序列化格式,而 AMF0 则用作旧版的
ActionScript 1.0 和 2.0 的序列化格式。 在网络传输数据方面, AMF3 比 AMF0 更有效率。 AMF3 能将 int 和
uint 对象作为整数( integer )传输,并且能序列化 ActionScript 3.0 才支持的数据类型 , 比如
ByteArray , XML 和 Iexternalizable 。
AMF 很好的解决了内容的丰富性。(具体 AMF 格式参考附件格式文档)
3.3.1 AMF中的数据类型Data Types
AMF0 supports the following data types (with their type field values):
NUMBER = 0x00
BOOLEAN = 0x01
STRING = 0x02
OBJECT = 0x03
MOVIECLIP
= 0x04
NULL_VALUE = 0x05
UNDEFINED = 0x06
REFERENCE = 0x07
ECMA_ARRAY
= 0x08
OBJECT_END = 0x09
STRICT_ARRAY = 0x0a
DATE = 0x0b
LONG_STRING
= 0x0c
UNSUPPORTED = 0x0d
RECORD_SET = 0x0e
XML_OBJECT =
0x0f
TYPED_OBJECT = 0x10
Binary Format
AMF format for a value/object
consists of a type byte (see above) followed by zero or more bytes. This
section describes the bytes following the type byte for various types.
NUMBER (type byte: 0x00)
Numbers are stored as 8
byte (big endian) float double. On x86 you can just byteswap a double
to encode it correctly.
BOOLEAN (type byte: 0x01)
A boolean is encoded
in one byte. FIXME: is true sent as 0xff? 0x01?
STRING (type byte: 0x02)
A string is encoded as a
2 byte (big endian) count (number of bytes) followed by that many bytes
of text. Note: there is no null terminator.
I think the text is assumed to be UTF-8. Can someone double check me on this?
NULL_VALUE (type byte: 0x05)
A null has zero
bytes following the type byte
UNDEFINED (type byte: 0x06)
A undefined has zero
bytes following the type byte
OBJECT (type byte: 0x08)
An object is encoded as
a series of key/value pairs. The key is encoded as a STRING (above)
WITH NO TYPE BYTE, and the value is any AMF value.
The object encoding is terminated by 0x000009 (that is a zero length string key, followed by the OBJECT_END type byte described below.
OBJECT_END (type byte: 0x09)
This is not really a
value, but a marker for the end of an OBJECT. See above.
STRICT_ARRAY (type byte: 0x0a)
This is the
encoding for arrays such as ["foo", "bar", 1, 2, 3]. For a hash (a set
of key/value pairs) you'll need to use OBJECT above.
An array is encoded as 4 byte (big endian) integer which is the number of elements in the array, followed by that many AMF values.
That's it. There's no terminator of any kind.
Use in shared object files
While most AMF
objects are just a value, there is a special variation used by shared
object files for properties. Rather than start with the type field,
followed by the length, it starts with a byte count, then the name, and
then the regular AMF type field, the length, and then the data.
3.4 客户端和服务器的连接过程
3.4.1客户和服务器的握手
Flash Player 以系统时间作为种子通过某种算法生成的数字签名,大小是 1537 字节向服务器发起第一次握手,服务器根据客户端的数字签名产生一个 3073 字节的验证包,给客户端,客户端在接受到服务器的回应以后会发送一个 1536 字节的回复。
具体的流程:
发送第一次握手包 handshark1
接收第二次握手包 handshark2
发送的三次握手包 handshark3
第
一个握手包 handshark1 和服务器的回复握手包 handshark2 都是以 0X03 开头。这三次握手不是 RTMP
协议本身的内容,所以在这并没有包含 RTMP 的协议头。是服务器的厂家自己产品做验证用的,严格的说就是你必须用 Adobe
的客户端和服务器才能使用我的协议。
3.4.2客户和服务器通信
具体连接和请求视频的过程
发送 rtmp_connect 命令
发送本地带宽消息 . 默认是 125000
服务器返回服务器带宽信息
服务器
返回本地带宽信息
服务器返回连接成功消息 "NetConnection.Connect.Success"
客户端发送创建流请求
encodeCreateStreamPacket
服务器返回创建流成功消息
客户端发送播放文件消息 Rtmp_Play
服
务器返回 TYPE_CHUNK_SIZE 消息
服务器返回开始播放消息 "NetStream.Play.Start"
服务器返回
视频信息 (TYPE_STREAM_METADATA) ,包括大小,宽高,速率等等信息--文件长度可以在这里推算出来
RTMP
的净核决定了内容服务, adobe 的服务器采用的 AMF 格式的字串命令来控制视频的传输和播放,具体的字串命令信息如下:
(注:字串的定义有厂家( adobe )自己定义,只要满足 AMF 的格式就可以)
NetConnection.Call.Failed
NetConnection.Call.BadVersion
NetConnection.Connect.AppShutdown
NetConnection.Connect.Closed
NetConnection.Connect.Rejected
NetConnection.Connect.Success
NetStream.Clear.Success
NetStream.Clear.Failed
NetStream.Publish.Start
NetStream.Publish.BadName
NetStream.Failed
NetStream.Unpublish.Success
NetStream.Record.Start
NetStream.Record.NoAccess
NetStream.Record.Stop
NetStream.Record.Failed
NetStream.Play.InsufficientBW
NetStream.Play.Start
NetStream.Play.StreamNotFound
NetStream.Play.Stop
NetStream.Play.Failed
NetStream.Play.Reset
NetStream.Play.PublishNotify
NetStream.Play.UnpublishNotify
NetStream.Data.Start
Application.Script.Error
Application.Script.Warning
Application.Resource.LowMemory
Application.Shutdown
Application.GC
Play
Pause
demoService.getListOfAvailableFLVs
getStreamLength
connect
app
flashVer
swfUrl
tcUrl
fpad
capabilities
audioCodecs
audioCodecs
videoCodecs
videoFunction
pageUrl
createStream
deleteStream
duration
framerate
audiocodecid
audiodatarate
videocodecid
videodatarate
height
width
3.4.2数据的萃取
在服务器返回开始播放消息
"NetStream.Play.Start"
之后,服务器就会开始给客户端传输数据了,一般数据的萃取都是先解析协议的头,然后根据协议头中数据类型和净核长度就可以把数据部分取出, RTMP
协议也是这样。
struct RTMP_HEAD
{
char cChannelid : 6;// 第一个字节的后 6 位
char cCheadsize ; // 第一个字节的头两位
char cTimer[3]; // 三个字节表示的时间信息
char cLength[3]; // 三个字节表示的长度
char cDatatype; // 数据类型
char sStreamid[4]; // 流标识
}
首先判断 cDatatype 是那种类型,然后根据不同的类型进行萃取数据部分,进行不同的处理,获取视频的数据的方式先看是否是一下的类型:
0×08 Audio Data packet containing audio
0×09 Video Data
packet containing video data
根据净核的长度读取出内存中的音视频数据,这里的音视频数据是有一定编码格式的数据,这个取决于应用的具体配置, Flash play
使用的是 FLV 的格式。要对这部分数据进行存取,还有做一部分工作,对 FLV 的视频数据进行去壳,取出数据保存文件就可以了。
------------------------------------------------------------------------------------
以下是百度百科上转载过来的,方便与上文对照理解:
rtmp
在 AppleTalk 协议组中,路由选择表维护协议(RTMP,Routing Table Protocol)是一种传输层协议,它在 AppleTalk 路由器中建立并维护路由选择表。RTMP 基于路由选择信息协议(RIP)。正如 RIP 一样,RTMP 使用跳数作为路由计量标准。一个数据包从源 网络发送到目标网络,必须通过的路由器或其它中间介质节点数目的计算结果即为跳数。
Real Time Messaging Protocol(实时消息传送协议协议)概述
实时消息传送协议是Adobe Systems公司为Flash播放器和服务器之间音频、视频和数据传输开发的私有协议。它有三种变种:
1)工作在TCP之上的明文协议,使用端口1935;
2)RTMPT封装在HTTP请求之中,可穿越防火墙;
3)RTMPS类似RTMPT,但使用的是HTTPS连接;
介绍:
RTMP协议是被Flash用于对象,视频,音频的传输.该协议建立在TCP协议或者轮询HTTP协议之上.
RTMP协议就像一个用来装数据包的容器,这些数据可以是AMF格式的数据,也可以是FLV中的视/音频数据.
一个单一的连接可以通过不同的通道传输多路网络流.这些通道中的包都是按照固定大小的包传输的.
网络连接(Connection)
一个Actionscript连接并播放一个流的简单代码:
复制内容到剪贴板代码:
var videoInstance:Video = your_video_instance;
var nc:NetConnection = new NetConnection();
var connected:Boolean = nc.connect("rtmp:/localhost/myapp");
var ns:NetStream = new NetStream(nc);
videoInstance.attachVideo(ns);
ns.play("flvName");
默认端口为1935
握手
Client → Server :向服务器发出握手请求.这不属于协议包一部分,该握手请求第一个字节为(0×03),其后跟着1536个字节.经管看上去这部分的内容对于RTMP协议 来说并不是至关重要的,但也不可随意对待.
Server → Client :服务器向客户端回应握手请求.这部分的数据仍然不属于RTMP协议的部分.该回应的其实字节仍然为(0x03),但是后边跟着个长度为1536个字节 (一共为3072 )的包块.第一个1536块看上去似乎可以是任意内容,甚至好像可以是Null都没有关系.第二个1536的代码块,是上一步客户端向服务器端发送的握手 请求的内容.
Client→Server:把上一步服务器向客户端回应的第二块1536个字节的数据块.
至此客户端与服务器端的握手结束,下面将发送RTMP协议的包内容.
Client → Server :向服务器发送连接包.
Server → Client :服务器回应.
... .... 等等... ...
RTMP 数据类型
0×01 Chunk Size changes the chunk size for packets
0×02 Unknown anyone know this one?
0×03 Bytes Read send every x bytes read by both sides
0×04 Ping ping is a stream control message, has subtypes
0×05 Server BW the servers downstream bw
0×06 Client BW the clients upstream bw
0×07 Unknown anyone know this one?
0×08 Audio Data packet containing audio
0×09 Video Data packet containing video data
0x0A - 0×11 Unknown anyone know?
0×12 Notify an invoke which does not expect a reply
0×13 Shared Object has subtypes
0×14 Invoke like remoting call, used for stream actions too.
Shared Object 数据类型
0×01 Connect
0×02 Disconnect
0×03 Set Attribute
0×04 Update Data
0×05 Update Attribute
0×06 Send Message
0×07 Status
0×08 Clear Data
0×09 Delete Data
0x0A Delete Attribute
0x0B
Initial Data
RTMP包结构
RTMP包 包含一个固定长度的包头和一个最长为128字节的包体.包头可以是下面4种长度的任意一种:12, 8, 4, or 1 byte(s).
第一个字节的前两个Bit很重要,它决定了包头的长度.它可以用掩码0xC0进行"与"计算.下面的表格罗列了可能的包头长度:Bits Header Length
00 12 bytes
01 8 bytes
10 4 bytes
11 1 byte
我们在这里讨论关RTMP包结构的问题并不是非常的详细.我们在以后有时间会讨论关于AMF的问题(敬请期待...:loveliness:),其实 RTMP包结构就是使用了AMF格式.
流
关于流的操作我们需要进一步研究,在论坛中的http://www.openred5.com/bbs /viewthread.php?tid=175&extra=page%3D1这篇文章研究的还是不错的,大家可以参考.不过下面可以列一个关 于客户端向服务器端发送流的流程:
Client→Server :发送一个创建流的请求.
Server→Client :返回一个表示流的索引号.
Client→Server :开始发送.
Client→Server :发送视音频数据包(这些包在同一个频道(channel)并用流的索引号来唯一标识).
发表评论
-
Speex vs Nellymoser
2011-02-21 10:42 2100Speex vs Nellymoser License ty ... -
Screen Video V1 Codec Format (SVC1) 和Screen Video V2 Codec Format(SVC2)
2011-01-25 15:01 2122SVC2 比 SVC1在压缩效率上有了很大的提升,相比SVC1 ... -
基于Flex的Flash聊天室的实现
2011-01-25 13:52 26641.简要介绍 用Red5有段时间了,知识总是零零碎碎的, ... -
Adobe Flash Screen Video bit stream format(SVC1)的介绍
2011-01-17 13:10 19461.Adobe Flash Screen Video bit ... -
在Red5 0.9版本实现 的上行带宽检测
2010-12-29 21:12 3664一、为什么要带宽检测 ... -
Flex中判断单击还是双击的方法
2010-08-27 06:45 1778Flex中,在双击事件时,单击事件也会同时相应,需要解决双击时 ...
相关推荐
描述中提到论文详细探讨了RTP协议到RTMP协议转换的实现方法和其意义。在实际应用中,可能会遇到需要将RTP协议数据流转换为RTMP格式以适应特定的网络环境或平台,例如为了在Web环境中嵌入或播放使用RTP传输的流媒体...
"rtmp协议文档.7z"这个压缩包包含了一系列关于RTMP协议的学习资料,包括中文翻译,对于想要深入理解RTMP的人来说是必不可少的资源。 首先,我们来看看"rtmp规范翻译1.0.docx",这可能是RTMP协议官方规范的中文翻译...
以下是RTMP协议的核心知识点,基于其交互过程和报文构成的详细说明: 1. **协议概述**: RTMP协议主要用于在服务器和客户端之间进行音视频数据的实时传输,如直播或点播服务。它基于TCP,提供可靠的连接,并且支持...
RTMP(Real-Time Messaging Protocol,实时消息传输协议)是一种基于TCP的应用层协议,主要用于在...通过研究提供的源代码,开发者可以深入理解RTMP协议的实现细节,并可能对其进行优化或扩展,以满足特定应用需求。
本项目“rtmpServer-master_nettyrtmp_rtmp推流_netty开发rtmp_rtmpServer-master”是针对RTMP协议开发的一个服务器端实现,它基于强大的Java网络库Netty。以下将详细介绍该服务器的开发背景、功能、核心技术和应用...
本项目"VideoSender.rar"就是一个典型的应用实例,它巧妙地结合了DirectShow和RTMP协议,实现了从摄像头采集视频并实时发送到RTMP服务器的功能。下面将详细解析这个项目的知识点。 首先,DirectShow是Microsoft提供...
再者,RTMP(Real-Time Messaging Protocol)协议的运用,是将编码后的H264流传输到rtmp服务器的关键。RTMP是一种专为音视频传输设计的协议,它允许客户端与服务器之间建立长连接,实现实时的数据交换。在这个项目中...
通过研究librtmp的源代码,我们可以了解RTMP协议的工作原理,包括连接建立、握手过程、数据通道的建立与管理,以及如何处理RTMP命令和数据包。 在rtmpdump源码中,我们可以看到以下几个关键部分: 1. `rtmp.h`: ...
通过捕获的流量,我们可以分析视频和音频编码类型、帧率、分辨率等信息,这对于优化流媒体服务、调试播放问题或者研究RTMP协议本身非常有用。 此外,`rtmpdump`也常用于制作离线版本的流媒体内容,例如制作直播回放...
《深入理解RTMP协议与LiteOS在物联网中的应用》 RTMP(Real-Time Messaging Protocol)是一种广泛应用于在线流媒体传输的协议,它由Adobe Systems开发,主要用于实时音频、视频和数据的传输。RTMP协议主要在服务器...
2. **RTMP协议解析**:源代码会解析RTMP协议的固定头部、命令消息和数据消息,理解不同类型的AMF(Action Message Format)编码。 3. **文件写入**:rtmpdump将接收到的数据流写入本地文件,实现流媒体的录制或保存...
专题报告 RTMP 协议 中文文档,有研究此方向的可以看看
《rtmpdump:深入理解RTMP协议与直播流媒体技术》 在当今互联网时代,视频直播已经成为人们日常生活的一部分,无论是在线教育、娱乐直播还是企业会议,流媒体技术都发挥着至关重要的作用。rtmpdump是一款强大的开源...
RTMP协议通过TCP提供服务,能够实现低延迟的音视频流传输,广泛应用于在线直播、视频会议等领域。而RTMPE是RTMP的增强版,主要增加了数据的安全性。 RTMPE协议在RTMP的基础上加入了加密机制,确保了数据在传输过程...
而RTMPDump则是开发者研究、调试和分析RTMP协议的首选工具。 这个项目是基于"rtmpdump 2.4"版本,通过Microsoft Visual C++ 6.0(简称VC6.0)进行编译的。VC6.0虽然是一款较老的开发环境,但它依然被许多开发者用于...
总结来说,这个"RTMP.zip"压缩包文件提供的是一种RTMP客户端的实现,特别关注AMF数据的解析,对于理解RTMP协议、开发音视频流应用或进行相关研究具有很高的价值。开发者可以通过这个实现深入学习RTMP协议的工作原理...
通过研究rtmpdump的源代码,开发者不仅可以学习RTMP协议的工作原理,还能掌握C++在实际项目中的应用,这对于开发流媒体相关的应用或者进行网络编程都是宝贵的资源。同时,这个项目也可以作为教学示例,帮助初学者...
首先,让我们来理解RTMP协议。RTMP是一种专为实时数据传输设计的协议,主要用于音频、视频数据的流式传输。当从RTMP服务器接收流时,我们需要将其保存为本地文件,以便离线观看或进一步处理。 1. **librtmp实现FLV*...