100行代碼實現最簡單的基於FFMPEG+SDL的視頻播放器

FFMPEG工程浩大,能夠參考的書籍又不是不少,所以不少剛學習FFMPEG的人經常感受到無從下手。我剛接觸FFMPEG的時候也感受不知從何學起。ide

所以我把本身作項目過程當中實現的一個很是簡單的視頻播放器(大約100行代碼)源代碼傳上來,以做備忘,同時方便新手學習FFMPEG。學習

該播放器雖然簡單,可是幾乎包含了使用FFMPEG播放一個視頻全部必備的API,而且使用SDL顯示解碼出來的視頻。ui

而且支持流媒體等多種視頻輸入,處於簡單考慮,沒有音頻部分,同時視頻播放採用直接延時40ms的方式spa

平臺使用VC2010.net

使用了最新的FFMPEG類庫code

直接貼代碼orm

int _tmain(int argc, _TCHAR* argv[])
{
	AVFormatContext	*pFormatCtx;
	int				i, videoindex;
	AVCodecContext	*pCodecCtx;
	AVCodec			*pCodec;
	char filepath[]="nwn.mp4";
	av_register_all();
	avformat_network_init();
	pFormatCtx = avformat_alloc_context();
	if(avformat_open_input(&pFormatCtx,filepath,NULL,NULL)!=0){
		printf("沒法打開文件\n");
		return -1;
	}
	if(av_find_stream_info(pFormatCtx)<0)
	{
		printf("Couldn't find stream information.\n");
		return -1;
	}
	videoindex=-1;
	for(i=0; i<pFormatCtx->nb_streams; i++) 
		if(pFormatCtx->streams[i]->codec->codec_type==AVMEDIA_TYPE_VIDEO)
		{
			videoindex=i;
			break;
		}
		if(videoindex==-1)
		{
			printf("Didn't find a video stream.\n");
			return -1;
		}
		pCodecCtx=pFormatCtx->streams[videoindex]->codec;
		pCodec=avcodec_find_decoder(pCodecCtx->codec_id);
		if(pCodec==NULL)
		{
			printf("Codec not found.\n");
			return -1;
		}
		if(avcodec_open(pCodecCtx, pCodec)<0)
		{
			printf("Could not open codec.\n");
			return -1;
		}
		AVFrame	*pFrame,*pFrameYUV;
		pFrame=avcodec_alloc_frame();
		pFrameYUV=avcodec_alloc_frame();
		uint8_t *out_buffer;
		out_buffer=new uint8_t[avpicture_get_size(PIX_FMT_YUV420P, pCodecCtx->width, pCodecCtx->height)];
		avpicture_fill((AVPicture *)pFrameYUV, out_buffer, PIX_FMT_YUV420P, pCodecCtx->width, pCodecCtx->height);
//------------SDL----------------
		if(SDL_Init(SDL_INIT_VIDEO | SDL_INIT_AUDIO | SDL_INIT_TIMER)) {  
			printf( "Could not initialize SDL - %s\n", SDL_GetError()); 
			exit(1);
		} 
		SDL_Surface *screen; 
		screen = SDL_SetVideoMode(pCodecCtx->width, pCodecCtx->height, 0, 0);
		if(!screen) {  printf("SDL: could not set video mode - exiting\n");  
		exit(1);
		}
		SDL_Overlay *bmp; 
		bmp = SDL_CreateYUVOverlay(pCodecCtx->width, pCodecCtx->height,SDL_YV12_OVERLAY, screen); 
		SDL_Rect rect;
//---------------
		int ret, got_picture;
		static struct SwsContext *img_convert_ctx;
		int y_size = pCodecCtx->width * pCodecCtx->height;

		AVPacket *packet=(AVPacket *)malloc(sizeof(AVPacket));
		av_new_packet(packet, y_size);
		//輸出一下信息-----------------------------
		printf("文件信息-----------------------------------------\n");
		av_dump_format(pFormatCtx,0,filepath,0);
		printf("-------------------------------------------------\n");
		//------------------------------
		while(av_read_frame(pFormatCtx, packet)>=0)
		{
			if(packet->stream_index==videoindex)
			{
				ret = avcodec_decode_video2(pCodecCtx, pFrame, &got_picture, packet);
				if(ret < 0)
				{
					printf("解碼錯誤\n");
					return -1;
				}
				if(got_picture)
				{
					img_convert_ctx = sws_getContext(pCodecCtx->width, pCodecCtx->height, pCodecCtx->pix_fmt, pCodecCtx->width, pCodecCtx->height, PIX_FMT_YUV420P, SWS_BICUBIC, NULL, NULL, NULL); 
					sws_scale(img_convert_ctx, (const uint8_t* const*)pFrame->data, pFrame->linesize, 0, pCodecCtx->height, pFrameYUV->data, pFrameYUV->linesize);

					SDL_LockYUVOverlay(bmp);
					bmp->pixels[0]=pFrameYUV->data[0];
					bmp->pixels[2]=pFrameYUV->data[1];
					bmp->pixels[1]=pFrameYUV->data[2];     
					bmp->pitches[0]=pFrameYUV->linesize[0];
					bmp->pitches[2]=pFrameYUV->linesize[1];   
					bmp->pitches[1]=pFrameYUV->linesize[2];
					SDL_UnlockYUVOverlay(bmp); 
					rect.x = 0;    
					rect.y = 0;    
					rect.w = pCodecCtx->width;    
					rect.h = pCodecCtx->height;    
					SDL_DisplayYUVOverlay(bmp, &rect); 
					//延時40ms
					SDL_Delay(40);
				}
			}
			av_free_packet(packet);
		}
		delete[] out_buffer;
		av_free(pFrameYUV);
		avcodec_close(pCodecCtx);
		avformat_close_input(&pFormatCtx);

		return 0;
}

軟件運行截圖: 視頻

完整工程下載地址:get

http://download.csdn.net/detail/leixiaohua1020/5122959 input

相關文章
相關標籤/搜索