资源下载: http://download.csdn.net/source/444512 实时流协议RTSP(RealTimeStreamingProtocol)是由RealNetworks和 Netscape共同提出的,该协议定义了一对多应用程序如何有效地通过IP网络传送多媒体数据。RTSP在体系结构上位于RTP(实时传输)和RTCP(实时控制)之上,它使用 TCP或RTP完成数据传输。HTTP与RTSP相比,HTTP传送HTML,而RTP传送的是多媒体数据。HTTP请求由客户机发出,服务器作出响应;使用RTSP时,客户机和服务器都可以发出请求,即RTSP可以是双向的。 实时流协议(RTSP)是应用级协议,控制实时数据的发送。RTSP提供了一个可扩展框架,使实时数据,如音频与视频,的受控、点播成为可能。 RTSP是应用级的协议,完成多媒体服务器的远程控制,控制信息的传输可以使用TCP,控制指令包括如:Setup、Play、Record、Pause、Teardwon等等。 对于流媒体应用,用户和服务器都可以发出请求,请求包括几种连接方法:持久、每个请求/响应传输一个连接、无连接。 常见的URL流媒体地址如: rtsp://media.example.com:554 RTP 数据报组成:Header + Payload RTCP:应用程序启动RTP会话时将同时占用两个端口,供RTP和RTCP使用。 如果有必要,RTP使用时可以有两个伴随文档:1)配置文档,定义负载的编码类型和格式。2)负载格式的规范文档。 在流传输过程中,有两类服务完成对流的转发处理: 1)译流服务器Translator,进入的流在流出时发生变化,作用之一是更好地穿越防火墙。 2)混流服务器Mixer,多个流进入,合并后变成一个流流出。 由于进入的流可能有多个源,比如视频会议,会有多个话筒和视频头等等情况,对于RTP来说,就有一个同步化源的问题,因此,RTP协议中用SSRC(Synchronization Source)字段来供Mixer实现同步功能。 Translator的一个作用是多播变成多个单播。 为了提供播放和回放功能,RTP提供时间标签+***,在流动的概念中,时间标签是最重要的信息。 RTP报文不提供长度和报文边界的描述。 RTP虽然是传输层协议,但没有在OSI体系中作为单独的层来使用。 RTP是目前解决流媒体实时传输问题的最好办法,如果要开发,可以选择JRTPLIB库。JRTPLIB是一个面向对象的RTP库,它完全遵循RFC 1889设计。JRTPLIB是一个用C++语言实现的RTP库,目前已经可以运行在Windows、Linux、FreeBSD、Solaris、Unix和 VxWorks等多种操作系统上。 了解更多RTP参考: http://www.cnitblog.com/zouzheng/archive/2008/01/04/38449.html 下面的例子参考jrtplib的example1,加了解析负载的部分。
// RTPClient.cpp:Definestheentrypointfortheconsoleapplication. // #include " stdafx.h " #include" rtpsession.h " #include" rtppacket.h " #include" rtpudpv4transmitter.h " #include" rtpipv4address.h " #include" rtpsessionparams.h " #include" rtperrors.h " #include< winsock2.h > #include< stdlib.h > #include< stdio.h > #include" windows.h " #include< iostream > #include< string > using namespace std; #pragma comment(lib,"jrtplib.lib") #pragma comment(lib,"jthread.lib") #pragma comment(lib,"WS2_32.lib") void checkerror( int rtperr) ... { if (rtperr < 0 ) ... { std::cout << " ERROR: " << RTPGetErrorString(rtperr) << std::endl; exit( - 1 ); } }int main( int argc, char * argv[]) ... { #ifdefWIN32 WSADATAdat; WSAStartup(MAKEWORD( 2 , 2 ), & dat); #endif // WIN32 RTPSessionsess; uint16_tportbase,destport; uint32_tdestip; std::string ipstr; int status,i,num; BYTE * pBuffer; BYTE * pfBuffer; // 输入一些必要信息 std::cout << " Enterlocalportbase: " << std::endl; std::cin >> portbase; std::cout << std::endl; std::cout << " EnterthedestinationIPaddress " << std::endl; std::cin >> ipstr; destip = inet_addr(ipstr.c_str()); if (destip == INADDR_NONE) ... { std::cerr << " BadIPaddressspecified " << std::endl; return - 1 ; } destip= ntohl(destip); std::cout << " Enterthedestinationport " << std::endl; std::cin >> destport; std::cout << std::endl; std::cout << " Numberofpacketsyouwishtobesent: " << std::endl; std::cin >> num; // 创建RTPsession RTPUDPv4TransmissionParamstransparams; RTPSessionParamssessparams; // IMPORTANT:ThelocaltimestampunitMUSTbeset,otherwise // RTCPSenderReportinfowillbecalculatedwrong // Inthiscase,we'llbesending10sampleseachsecond,sowe'll // putthetimestampunitto(1.0/10.0) sessparams.SetOwnTimestampUnit( 1.0 / 10.0 ); sessparams.SetAcceptOwnPackets( true ); transparams.SetPortbase(portbase); status = sess.Create(sessparams, & transparams); checkerror(status); RTPIPv4Addressaddr(destip,destport); status = sess.AddDestination(addr); checkerror(status); for (i = 1 ;i <= num;i ++ ) ... { printf( " Sendingpacket%d/%d " ,i,num); // 发送数据“1234567890” status = sess.SendPacket(( void * ) " 1234567890 " , 10 , 0 , false , 10 ); checkerror(status); sess.BeginDataAccess(); // checkincomingpackets if (sess.GotoFirstSourceWithData()) ... { do ... { RTPPacket * pack; while ((pack = sess.GetNextPacket()) != NULL) ... { // Youcanexaminethedatahere printf( " Gotpacket! " ); std::cout << " Gotpacketwith " << " extendedsequencenumber " << pack -> GetExtendedSequenceNumber() << " fromSSRC " << pack -> GetSSRC() << std::endl; int dataLength = pack -> GetPayloadLength(); pfBuffer = (unsigned char * )pack -> GetPayloadData(); pBuffer = new BYTE[dataLength + 1 ]; memcpy(pBuffer,pfBuffer,dataLength); pBuffer[dataLength] = 0 ; std::cout << pBuffer << std::endl; // wedon'tlongerneedthepacket,so // we'lldeleteit sess.DeletePacket(pack); } }while (sess.GotoNextSourceWithData()); } sess.EndDataAccess(); #ifndefRTP_SUPPORT_THREAD status= sess.Poll(); checkerror(status); #endif // RTP_SUPPORT_THREAD RTPTime::Wait(RTPTime(1 , 0 )); } sess.BYEDestroy(RTPTime(10 , 0 ), 0 , 0 ); #ifdefWIN32 WSACleanup(); #endif // WIN32 return 0 ; }
编译注意修改每个Source File的code generation下的Use run-time library为Debug Multithreaded DLL。 如图 关于jrtplib的环境,可以参考网上很多的资料,也可以从我的资源里下载,我已经编译好了相关lib,只要加的VC环境里就可以了。 执行测试程序的效果如下: Enter local portbase: 8000 Enter the destination IP address 127.0.0.1 Enter the destination port 8000 Number of packets you wish to be sent: 5 Sending packet 1/5 Got packet ! Got packet with extended sequence number 59262 from SSRC 3029241192 1234567890 Sending packet 2/5 Got packet ! Got packet with extended sequence number 59263 from SSRC 3029241192 1234567890 Sending packet 3/5 Sending packet 4/5 Got packet ! Got packet with extended sequence number 59264 from SSRC 3029241192 1234567890 Got packet ! Got packet with extended sequence number 59265 from SSRC 3029241192 1234567890 Sending packet 5/5 上面执行的意思是程序自己开了8000端口,然后往自己的8000发送,所以不仅发送出去,还收到并解析出了内容。如果要往另外机器上发,另一个机器上也运行这个程序就可以了。当然可以专门再写一个接收端。
相关文章: