使用librtmp库发布直播流

本文介绍如何使用RTMP协议与FLV文件格式实现视频的实时传输。通过解析FLV文件,并利用RTMP协议将其发送到服务器,最终实现实时流媒体的播放效果。文中提供了一个C语言实现的具体示例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

rtmp协议与flv封包格式基本相同(本来就是一家公司的嘛)
将flv文件解析出类型,时间戳,数据大小,数据区
对应写的rtmp包的位置进行发送即可

这里是以flv文件做源,可以自己采集视频音频进行实时发送
库版本 2.4

下图,左边发送到服务器,右侧可正常观看(模拟的是实时流,发完该流也就结束了)
使用rtmp库将flv文件发布直播流 - 幻想少佳 - 幻想少佳的博客
关键函数
RTMP_EnableWrite(rtmp);
将rtmp设置可写状态,会发出publish指令,否则是play指令
#include <stdio.h>
#include <stdlib.h>
#include <time.h>
#include "librtmp/rtmp_sys.h"
#include "librtmp/log.h"

#pragma comment(lib,"ws2_32.lib")
#pragma comment(lib,"lib/librtmp.lib")
//网络与本地字节转换
#define HTON16(x)  ((x>>8&0xff)|(x<<8&0xff00))
#define HTON24(x)  ((x>>16&0xff)|(x<<16&0xff0000)|x&0xff00)
#define HTON32(x)  ((x>>24&0xff)|(x>>8&0xff00)|\
(x<<8&0xff0000)|(x<<24&0xff000000))

bool Read8(int &i8,FILE*fp);
bool Read16(int &i16,FILE*fp);
bool Read24(int &i24,FILE*fp);
bool Read32(int &i32,FILE*fp);
bool Peek8(int &i8,FILE*fp);
bool ReadTime(int &itime,FILE*fp);

//RTMP_XXX()返回0表示失败,1表示成功
RTMP*rtmp=NULL;//rtmp应用指针
RTMPPacket*packet=NULL;//rtmp包结构
char* rtmpurl="rtmp://127.0.0.1:1935/live/test";//连接的URL
char* flvfile="test.flv";//读取的flv文件

bool ZINIT();//初始化相关
void ZCLEAR();//清除相关

int main()
{
	if (!ZINIT())
	{
		printf("init socket err\n");
		return -1;
	}
/////////////////////////////////初始化//////////////////////	
	RTMP_LogLevel lvl=RTMP_LOGINFO;
	RTMP_LogSetLevel(lvl);//设置信息等级
	//RTMP_LogSetOutput(FILE*fp);//设置信息输出文件
	rtmp=RTMP_Alloc();//申请rtmp空间
	RTMP_Init(rtmp);//初始化rtmp设置
	rtmp->Link.timeout=5;//设置连接超时,单位秒,默认30秒
	packet=new RTMPPacket();//创建包
	RTMPPacket_Alloc(packet,1024*64);//给packet分配数据空间,要满足最长的帧,不知道可设大些
	RTMPPacket_Reset(packet);//重置packet状态
////////////////////////////////连接//////////////////
	RTMP_SetupURL(rtmp,rtmpurl);//设置url
	RTMP_EnableWrite(rtmp);//设置可写状态
	if (!RTMP_Connect(rtmp,NULL))//连接服务器
	{
		printf("connect err\n");
		ZCLEAR();
		return -1;
	}
	if (!RTMP_ConnectStream(rtmp,0))//创建并发布流(取决于rtmp->Link.lFlags)
	{
		printf("ConnectStreamerr\n");
		ZCLEAR();
		return -1;
	}
	packet->m_hasAbsTimestamp = 0; //绝对时间戳
	packet->m_nChannel = 0x04; //通道
	packet->m_nInfoField2 = rtmp->m_stream_id;

	FILE*fp=fopen(flvfile,"rb");
	if (fp==NULL)
	{
		printf("open file:%s err\n",flvfile);
		ZCLEAR();
		return -1;
	}

	printf("rtmpurl:%s\nflvfile:%s\nsend data ...\n",rtmpurl,flvfile);
////////////////////////////////////////发送数据//////////////////////
	fseek(fp,9,SEEK_SET);//跳过前9个字节
	fseek(fp,4,SEEK_CUR);//跳过4字节长度
	long start=clock()-1000;
	long perframetime=0;//上一帧时间戳
	while(TRUE)
	{
		if((clock()-start)<perframetime)//发的太快就等一下
		{	
			Sleep(500);
			continue;
		}
		int type=0;//类型
		int datalength=0;//数据长度
		int time=0;//时间戳
		int streamid=0;//流ID
		if(!Read8(type,fp))
			break;
		if(!Read24(datalength,fp))
			break;
		if(!ReadTime(time,fp))
			break;
		if(!Read24(streamid,fp))
			break;

		if (type!=0x08&&type!=0x09)
		{
			fseek(fp,datalength+4,SEEK_CUR);
			continue;
		}

		if(fread(packet->m_body,1,datalength,fp)!=datalength)
			break;
		packet->m_headerType = RTMP_PACKET_SIZE_MEDIUM; 
		packet->m_nTimeStamp = time; 
		packet->m_packetType=type;
		packet->m_nBodySize=datalength;

		if (!RTMP_IsConnected(rtmp))
		{
			printf("rtmp is not connect\n");
			break;
		}
		if (!RTMP_SendPacket(rtmp,packet,0))
		{
			printf("send err\n");
			break;
		}
		int alldatalength=0;//该帧总长度
		if(!Read32(alldatalength,fp))
			break;
		perframetime=time;
	}
	printf("send data over\n");
	fclose(fp);
	ZCLEAR();
	return 0;
}

bool ZINIT()
{
	WORD version;
	WSADATA wsaData;
	version=MAKEWORD(2,2);
	if(WSAStartup(version,&wsaData)!=0)
	{
		return FALSE;
	}
	return TRUE;
}
void ZCLEAR()
{
	//////////////////////////////////////////释放/////////////////////
	if (rtmp!=NULL)
	{
		RTMP_Close(rtmp);//断开连接
		RTMP_Free(rtmp);//释放内存
		rtmp=NULL;
	}
	if (packet!=NULL)
	{
		RTMPPacket_Free(packet);//释放内存
		delete packet;
		packet=NULL;
	}
	///////////////////////////////////////////////////
	WSACleanup();
}


bool Read8(int &i8,FILE*fp)
{
	if(fread(&i8,1,1,fp)!=1)
		return false;
	return true;
}
bool Read16(int &i16,FILE*fp)
{
	if(fread(&i16,2,1,fp)!=1)
		return false;
	i16=HTON16(i16);
	return true;
}
bool Read24(int &i24,FILE*fp)
{
	if(fread(&i24,3,1,fp)!=1)
		return false;
	i24=HTON24(i24);
	return true;
}
bool Read32(int &i32,FILE*fp)
{
	if(fread(&i32,4,1,fp)!=1)
		return false;
	i32=HTON32(i32);
	return true;
}
bool Peek8(int &i8,FILE*fp)
{
	if(fread(&i8,1,1,fp)!=1)
		return false;
	fseek(fp,-1,SEEK_CUR);
	return true;
}
bool ReadTime(int &itime,FILE*fp)
{
	int temp=0;
	if(fread(&temp,4,1,fp)!=1)
		return false;
	itime=HTON24(temp);
	itime|=(temp&0xff000000);
	return true;
}


源码下载:(包含所需第三方库和播放器, 需自己开启rtmp服务器)
http://vdisk.weibo.com/s/unhDlwQMVR3Fv 
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值