ffmpeg音视频开发从入门到精通——ffmpeg日志及目录操作

文章目录

    • FFMPEG
    • [1. 操作日志](#1. 操作日志)
    • [2. 文件移动和删除](#2. 文件移动和删除)
    • [3. 操作目录重要函数](#3. 操作目录重要函数)

FFMPEG

1. 操作日志

  • 日志级别

AV LOG ERROR
AV LOG WARNING
AV LOG INFO
AV LOG DEBUG

cmake 复制代码
cmake_minimum_required(VERSION 3.27)
project(FFmpeg_exercise)
set(CMAKE_CXX_STANDARD 14)

# 定义FFmpeg的安装路径变量
set(FFMPEG_INSTALL_DIR "/usr/local/ffmpeg")

# 将FFmpeg的头文件目录添加到包含路径
include_directories(${FFMPEG_INSTALL_DIR}/include)

# 定义FFmpeg库的基础名称(根据你的需要调整)
set(FFMPEG_LIBS "avcodec;avformat;avutil") # 用分号分隔库名

# 寻找并链接FFmpeg库
foreach(FFMPEG_LIB ${FFMPEG_LIBS})
    find_library(${FFMPEG_LIB}_LIBRARY NAMES ${FFMPEG_LIB}
            PATHS ${FFMPEG_INSTALL_DIR}/lib NO_DEFAULT_PATH)
    list(APPEND FFMPEG_LIBRARIES ${${FFMPEG_LIB}_LIBRARY})
endforeach()

add_executable(FFmpeg_exercise main.cpp)
# 链接FFmpeg库
target_link_libraries(FFmpeg_exercise ${FFMPEG_LIBRARIES})
  • main.cpp
c++ 复制代码
#ifdef __cplusplus
extern "C" {
#endif
    // 包含FFmpeg的头文件
    #include <libavcodec/avcodec.h>
    #include <libavformat/avformat.h>
    #include <libavutil/avutil.h>
#ifdef __cplusplus

}
#endif
/*
 通过预处理器指令#ifdef __cplusplus来检查代码是否在C++环境中编译。
 如果是,它将开始一个extern "C"块,这意味着在该块内包含的代码将使用C的链接规则。
 这通常用于包含C库的头文件,以确保C++编译器不会对函数名进行名称修饰。
*/
int main() {
    av_log_set_level(AV_LOG_DEBUG);
    av_log(nullptr,AV_LOG_DEBUG,"Hello World!:%d\n",10);
    av_log(nullptr,AV_LOG_INFO,"Hello World! test: INFO\n");
    av_log(nullptr,AV_LOG_ERROR,"Hello World! test: ERROR\n");
    return 0;
}

2. 文件移动和删除

  • api

avpriv_io_delete(): 删除
avpriv_io_move():移动

  • 案例
cmake 复制代码
cmake_minimum_required(VERSION 3.27)
project(FFmpeg_exercise)
set(CMAKE_CXX_STANDARD 14)

# 定义FFmpeg的安装路径变量
set(FFMPEG_INSTALL_DIR "/usr/local/ffmpeg")

# 将FFmpeg的头文件目录添加到包含路径
include_directories(${FFMPEG_INSTALL_DIR}/include)

# 定义FFmpeg库的基础名称(根据你的需要调整)
set(FFMPEG_LIBS "avcodec;avformat;avutil") # 用分号分隔库名

# 寻找并链接FFmpeg库
foreach(FFMPEG_LIB ${FFMPEG_LIBS})
    find_library(${FFMPEG_LIB}_LIBRARY NAMES ${FFMPEG_LIB}
            PATHS ${FFMPEG_INSTALL_DIR}/lib NO_DEFAULT_PATH)
    list(APPEND FFMPEG_LIBRARIES ${${FFMPEG_LIB}_LIBRARY})
endforeach()

add_executable(FFmpeg_exercise main.cpp)
# 链接FFmpeg库
target_link_libraries(FFmpeg_exercise ${FFMPEG_LIBRARIES})
c++ 复制代码
#ifdef __cplusplus
extern "C" {
#endif
    // 包含FFmpeg的头文件
    #include <libavcodec/avcodec.h>
    #include <libavformat/avformat.h>
    #include <libavutil/avutil.h>
#ifdef __cplusplus

}
#endif
#include <iostream>
#include <fstream>
/*
 通过预处理器指令#ifdef __cplusplus来检查代码是否在C++环境中编译。
 如果是,它将开始一个extern "C"块,这意味着在该块内包含的代码将使用C的链接规则。
 这通常用于包含C库的头文件,以确保C++编译器不会对函数名进行名称修饰。
*/
int main() {
    std::ofstream outfile("./test.txt");
    outfile<<"hello world!\n"<<std::endl;
    outfile<<"this is a test file"<<std::endl;
    outfile.close();
    if (outfile.is_open()) {
        av_log(nullptr,AV_LOG_ERROR,"Failed to close file: %s","./test.txt");
        return 1; // 如果文件未关闭,返回错误代码
    }
    av_log_set_level(AV_LOG_DEBUG);

    int ret {0};
    ret = avpriv_io_move("./test.txt","./demo.txt");
    if (ret<0){
        av_log(nullptr,AV_LOG_ERROR,"Failed to rename text.txt\n");
    }
    av_log(nullptr,AV_LOG_INFO,"SUCCESSED TO RENAME FILE TEXT.TXT\n");
    ret = avpriv_io_delete("./demo.txt");
    if (ret<0){
        av_log(nullptr,AV_LOG_ERROR,"Failed to delete demo.txt\n");
    }
    av_log(nullptr,AV_LOG_INFO,"SUCCESSED TO delete FILE TEXT.TXT\n");


    //    av_log(nullptr,AV_LOG_DEBUG,"Hello World!:%d\n",10);
    //    av_log(nullptr,AV_LOG_INFO,"Hello World! test: INFO\n");
    //    av_log(nullptr,AV_LOG_ERROR,"Hello World! test: ERROR\n");
    return 0;
}

3. 操作目录重要函数

  1. avio_open_dir():

    • 功能:打开一个目录以供进一步读取。
    • 返回值:成功时返回指向AVIODirContext的指针,失败时返回NULL
    • 使用:此函数用于初始化目录遍历,获取指向目录中第一个条目的指针。
  2. AVIODirContext:

    • 类型:这是一个不透明的结构体,用于表示打开的目录和遍历状态。
    • 用途:在使用avio_open_dir()成功打开目录后,会返回一个AVIODirContext类型的指针,该指针在后续的目录遍历中使用。
  3. avio_read_dir():

    • 功能:读取目录中的下一个条目。
    • 参数:传入AVIODirContext指针。
    • 返回值:成功时返回指向AVIoDirEntry的指针,遍历完成或失败时返回NULL
    • 使用:在每次调用avio_read_dir()后,都会移动到目录中的下一个条目。如果已经到达目录末尾或发生错误,则返回NULL
  4. AVIoDirEntry:

    • 类型:这是一个结构体,包含有关目录条目的信息,如文件名、文件类型等。
    • 用途:通过avio_read_dir()获取的每个条目都是一个AVIoDirEntry对象,它包含了关于文件或目录的信息。
  5. avio_close_dir():

    • 功能:关闭之前用avio_open_dir()打开的目录。
    • 参数:传入AVIODirContext指针。
    • 返回值:返回0表示成功,返回负数表示失败。
    • 使用:完成目录遍历后,应当调用此函数来释放与AVIODirContext关联的资源。
  • 案例
c++ 复制代码
#ifdef __cplusplus
extern "C" {
#endif
// 包含FFmpeg的头文件
#include <libavcodec/avcodec.h>
#include <libavformat/avformat.h>
#include <libavutil/avutil.h>
#ifdef __cplusplus

}
#endif
#include <iostream>
#include <fstream>
/*
 通过预处理器指令#ifdef __cplusplus来检查代码是否在C++环境中编译。
 如果是,它将开始一个extern "C"块,这意味着在该块内包含的代码将使用C的链接规则。
 这通常用于包含C库的头文件,以确保C++编译器不会对函数名进行名称修饰。
*/
int main() {

    int ret {0};
    //# 上下文结构体
    AVIODirContext *ctx {nullptr};
    AVIODirEntry *entry {nullptr};
    ret = avio_open_dir(&ctx,"./", nullptr);
    if (ret<0){
        av_log(nullptr,AV_LOG_ERROR,"Failed to open dir:%s\n", av_err2str(ret));
        goto _fail;;
    } else{
        av_log(nullptr,AV_LOG_INFO,"Success to open dir:%s\n", av_err2str(ret));
    }

    while(true){
        // 读取文件
        ret = avio_read_dir(ctx,&entry);
        if (ret<0){
            av_log(nullptr,AV_LOG_ERROR,"Failed to read dir:%s\n", av_err2str(ret));
            return -1;
        } else{
            av_log(nullptr,AV_LOG_INFO,"Success to read dir:%s\n", av_err2str(ret));
        }
        if(!entry){
            break;
        }

        av_log(nullptr,AV_LOG_INFO,"%lld PRID64%s\n",entry->size,entry->name);
        // 释放内存,防止内存泄漏
        avio_free_directory_entry(&entry);
    }
    _fail:
    avio_close_dir(&ctx);

    std::ofstream outfile("./test.txt");
    outfile<<"hello world!\n"<<std::endl;
    outfile<<"this is a test file"<<std::endl;
    outfile.close();
    if (outfile.is_open()) {
        av_log(nullptr,AV_LOG_ERROR,"Failed to close file: %s","./test.txt");
        return 1; // 如果文件未关闭,返回错误代码
    }
    av_log_set_level(AV_LOG_DEBUG);

    ret = avpriv_io_move("./test.txt","./demo.txt");
    if (ret<0){
        av_log(nullptr,AV_LOG_ERROR,"Failed to rename text.txt\n");
    }
    av_log(nullptr,AV_LOG_INFO,"SUCCESSED TO RENAME FILE TEXT.TXT\n");
    ret = avpriv_io_delete("./demo.txt");
    if (ret<0){
        av_log(nullptr,AV_LOG_ERROR,"Failed to delete demo.txt\n");
    }
    av_log(nullptr,AV_LOG_INFO,"SUCCESSED TO delete FILE TEXT.TXT\n");


    //    av_log(nullptr,AV_LOG_DEBUG,"Hello World!:%d\n",10);
    //    av_log(nullptr,AV_LOG_INFO,"Hello World! test: INFO\n");
    //    av_log(nullptr,AV_LOG_ERROR,"Hello World! test: ERROR\n");

    return 0;
}
相关推荐
lxkj_202421 分钟前
使用线程局部存储解决ffmpeg中多实例调用下自定义日志回调问题
ffmpeg
runing_an_min5 小时前
ffmpeg视频滤镜:替换部分帧-freezeframes
ffmpeg·音视频·freezeframes
ruizhenggang5 小时前
ffmpeg本地编译不容易发现的问题 — Error:xxxxx not found!
ffmpeg
runing_an_min7 小时前
ffmpeg视频滤镜:提取缩略图-framestep
ffmpeg·音视频·framestep
韩曙亮19 小时前
【FFmpeg】FFmpeg 内存结构 ③ ( AVPacket 函数简介 | av_packet_ref 函数 | av_packet_clone 函数 )
ffmpeg·音视频·avpacket·av_packet_clone·av_packet_ref·ffmpeg内存结构
oushaojun21 天前
ubuntu中使用ffmpeg和nginx推流rtmp视频
nginx·ubuntu·ffmpeg·rtmp
莫固执,朋友1 天前
网络抓包工具tcpdump 在海思平台上的编译使用
网络·ffmpeg·音视频·tcpdump
lxkj_20241 天前
修改ffmpeg实现https-flv内容加密
网络协议·https·ffmpeg
cuijiecheng20181 天前
音视频入门基础:MPEG2-TS专题(6)——FFmpeg源码中,获取MPEG2-TS传输流每个transport packet长度的实现
ffmpeg·音视频
VisionX Lab1 天前
数据脱敏工具:基于 FFmpeg 的视频批量裁剪
python·ffmpeg·音视频