Ffmpeg 视频教程 向视频中添加文字
Ffmpeg支持添加文字功能,具体如何将文字叠加到视频中的每一张图片,FFmpeg调用了文字库FreeSerif.ttf。当我们
用到ffmpeg 添加文字功能时 我们需要先下载改文字库,下载地址是http://www.fonts2u.com/free-serif.font,这算是
前期准备工作。准备工作完成以后,我来介绍下Ffmpeg实现视频文件添加文字功能的基本流程,流程图如下图所示:
图1. Ffmpeg 向视频添加文字流程
Ffmpeg 以Filter的方式添加文字,下面给出具体的代码。
1. 打开视频文件。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 | int OpenInput( char *fileName) { context = avformat_alloc_context(); context->interrupt_callback.callback = interrupt_cb; AVDictionary *format_opts = nullptr ; int ret = avformat_open_input(&context, fileName, nullptr , &format_opts); if (ret < 0) { return ret; } ret = avformat_find_stream_info(context, nullptr ); av_dump_format(context, 0, fileName, 0); if (ret >= 0) { std::cout << "open input stream successfully" << endl; } return ret; } |
2. 初始化 初始化包括输出上下文以及Filter,解码器,编码器四个部分。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 | //初始化输出上下文 int OpenOutput( char *fileName) { int ret = 0; ret = avformat_alloc_output_context2(&outputContext, nullptr , "mpegts" , fileName); if (ret < 0) { goto Error; } ret = avio_open2(&outputContext->pb, fileName, AVIO_FLAG_READ_WRITE, nullptr , nullptr ); if (ret < 0) { goto Error; } for ( int i = 0; i < context->nb_streams; i++) { AVStream * stream = avformat_new_stream(outputContext, outPutEncContext->codec); stream->codec = outPutEncContext; if (ret < 0) { goto Error; } } av_dump_format(outputContext, 0, fileName, 1); ret = avformat_write_header(outputContext, nullptr ); if (ret < 0) { goto Error; } if (ret >= 0) cout << "open output stream successfully" << endl; return ret ; Error: if (outputContext) { avformat_close_input(&outputContext); } return ret ; } |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 | //初始化filter<br>int InitFilter(AVCodecContext * codecContext) { char args[512]; int ret = 0; AVFilter *buffersrc = avfilter_get_by_name( "buffer" ); AVFilter *buffersink = avfilter_get_by_name( "buffersink" ); AVFilterInOut *outputs = avfilter_inout_alloc(); AVFilterInOut *inputs = avfilter_inout_alloc(); string filters_descr = "drawtext=fontfile=D\\\\:FreeSerif.ttf:fontsize=100:text=hello world:x=100:y=100" ; enum AVPixelFormat pix_fmts[] = { AV_PIX_FMT_YUV420P, AV_PIX_FMT_YUV420P}; filter_graph = avfilter_graph_alloc(); if (!outputs || !inputs || !filter_graph) { ret = AVERROR(ENOMEM); goto end; } /* buffer video source: the decoded frames from the decoder will be inserted here. */ sprintf_s(args, sizeof (args), "video_size=%dx%d:pix_fmt=%d:time_base=%d/%d:pixel_aspect=%d/%d" , codecContext->width, codecContext->height, codecContext->pix_fmt, codecContext->time_base.num, codecContext->time_base.den, codecContext->sample_aspect_ratio.num, codecContext->sample_aspect_ratio.den); ret = avfilter_graph_create_filter(&buffersrc_ctx, buffersrc, "in" , args, NULL, filter_graph); if (ret < 0) { av_log(NULL, AV_LOG_ERROR, "Cannot create buffer source\n" ); goto end; } /* buffer video sink: to terminate the filter chain. */ ret = avfilter_graph_create_filter(&buffersink_ctx, buffersink, "out" , NULL, NULL, filter_graph); if (ret < 0) { av_log(NULL, AV_LOG_ERROR, "Cannot create buffer sink\n" ); goto end; } ret = av_opt_set_int_list(buffersink_ctx, "pix_fmts" , pix_fmts, AV_PIX_FMT_YUV420P, AV_OPT_SEARCH_CHILDREN); if (ret < 0) { av_log(NULL, AV_LOG_ERROR, "Cannot set output pixel format\n" ); goto end; } /* Endpoints for the filter graph. */ outputs->name = av_strdup( "in" ); outputs->filter_ctx = buffersrc_ctx; outputs->pad_idx = 0; outputs->next = NULL; inputs->name = av_strdup( "out" ); inputs->filter_ctx = buffersink_ctx; inputs->pad_idx = 0; inputs->next = NULL; if ((ret = avfilter_graph_parse_ptr(filter_graph, filters_descr.c_str(), &inputs, &outputs, NULL)) < 0) goto end; if ((ret = avfilter_graph_config(filter_graph, NULL)) < 0) goto end; return ret; end: avfilter_inout_free(&inputs); avfilter_inout_free(&outputs); return ret; } |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 | //初始化编码器<br>int InitEncoderCodec( int iWidth, int iHeight) { AVCodec * pH264Codec = avcodec_find_encoder(AV_CODEC_ID_H264); if (NULL == pH264Codec) { printf ( "%s" , "avcodec_find_encoder failed" ); return -1; } outPutEncContext = avcodec_alloc_context3(pH264Codec); outPutEncContext->gop_size = 30; outPutEncContext->has_b_frames = 0; outPutEncContext->max_b_frames = 0; outPutEncContext->codec_id = pH264Codec->id; outPutEncContext->time_base.num =context->streams[0]->codec->time_base.num; outPutEncContext->time_base.den = context->streams[0]->codec->time_base.den; outPutEncContext->pix_fmt = *pH264Codec->pix_fmts; outPutEncContext->width = iWidth; outPutEncContext->height = iHeight; outPutEncContext->me_subpel_quality = 0; outPutEncContext->refs = 1; outPutEncContext->scenechange_threshold = 0; outPutEncContext->trellis = 0; AVDictionary *options = nullptr ; outPutEncContext->flags |= AV_CODEC_FLAG_GLOBAL_HEADER; int ret = avcodec_open2(outPutEncContext, pH264Codec, &options); if (ret < 0) { printf ( "%s" , "open codec failed" ); return ret; } return 1; } <br> //初始化解码器 int InitDecodeCodec(AVCodecID codecId) { auto codec = avcodec_find_decoder(codecId); if (!codec) { return -1; } decoderContext = context->streams[0]->codec; if (!decoderContext) { fprintf (stderr, "Could not allocate video codec context\n" ); exit (1); } if (codec->capabilities & AV_CODEC_CAP_TRUNCATED) decoderContext->flags |= AV_CODEC_FLAG_TRUNCATED; int ret = avcodec_open2(decoderContext, codec, NULL); return ret; } |
如需交流,可以加QQ群1038388075,766718184,或者QQ:350197870
视频下载地址:http://www.chungen90.com/?list_53
Demo下载地址: http://www.chungen90.com/?list_52
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· AI技术革命,工作效率10个最佳AI工具