Window平台编译log4cpp使用方法记录 (一)

本文指导如何在VS2019环境下下载并解决log4cpp源码编译过程中出现的错误,涉及NTEventLogCategories.res问题及snprintf冲突处理,最终生成log4cpp.dll和lib文件。

1. 源码下载

先在网上下载源代码,点击:下载链接,下载后解压如下图,打开msvc10文件夹,源码是用vs2010打开,此处安装的是vs2019,打开会自动提示升级该项目,点击确定即可。
解压目录

2 右击log4cpp项目选择生成,首次编译会出现一些错误,解决方法如下:

错误1:无法打开输入文件“Debug\NTEventLogCategories.res”。
解决方法:①在log4cpp项目工程中找到NTEventLogCategories.mc文件,选择该文件上然后右键选择属性,在弹出窗口中找到“配置属性–>自定义生成工具–>常规–>命令行”中修改编译命令,设置为如下命令:

if not exist $(OutDir) md $(OutDir)
mc.exe -h $(OutDir) -r $(OutDir) $(ProjectDir)..\%(Filename).mc
RC.exe -r -fo $(OutDir)%(Filename).res $(OutDir)%(Filename).rc
link.exe /MACHINE:IX86 -dll -noentry -out:$(OutDir)NTEventLogAppender.dll $(OutDir)%(Filename).res

如图:
在这里插入图片描述

②在log4cppRunnable项目工程中找到NTEventLogCategories.mc文件,操作方法如①。

此时再编译工程,则会出现错误2.
错误2:函数"int snprintf(char* const,const size_t,const char*const,…)"已有主体的报警。
解决方法:由于log4cpp中对snprintf进行了重新实现,visual studio的c库对snprintf也有实现,windows中在链接时会报snprintf函数冲突,所以需要设置log4cpp的预编译项,选择使用visual stuido中c库的实现,在log4cpp工程上点右键选属性,在”配置属性–>C/C+±->预处理器–>预处理器定义"中增加一条预处理定义
添加预处理
此时再编译log4cpp工程,则不再有报警。
编译通过

3 此处选择的release版本编译,编译成功后在log4cpp和log4cppLIB两个工程编译目录release里分别产生log4cpp.dll和log4cppLIB.lib文件。

在这里插入图片描述

C:\Users\xy\Downloads\opencv-4.10.0\modules\highgui\src\window_w32.cpp(2194,5): error C2440: “static_cast”: 无法从“const c har [49]”转换为“const cv::utils::logging::LogTag *” [C:\Users\xy\Downloads\build\modules\highgui\opencv_highgui.vcxproj] (编译源文件“../../../opencv-4.10.0/modules/highgui/src/window_w32.cpp) C:\Users\xy\Downloads\opencv-4.10.0\modules\highgui\src\window_w32.cpp(2194,5): 指向的类型不相关; 转换需要 reinterpret_cast、C 样式强制转换或带圆括号的函数样式强制转换 C:\Users\xy\Downloads\opencv-4.10.0\modules\highgui\src\window_w32.cpp(2194,5): error C3536: “cv_temp_logtagptr”: 初始化之前 无法使用 [C:\Users\xy\Downloads\build\modules\highgui\opencv_highgui.vcxproj] (编译源文件“../../../opencv-4.10.0/modules/highgui/src/window_w32.cpp) C:\Users\xy\Downloads\opencv-4.10.0\modules\highgui\src\window_w32.cpp(2194,5): error C2059: 语法错误:“;” [C:\Users\xy\Down loads\build\modules\highgui\opencv_highgui.vcxproj] (编译源文件“../../../opencv-4.10.0/modules/highgui/src/window_w32.cpp) C:\Users\xy\Downloads\opencv-4.10.0\modules\highgui\src\window_w32.cpp(2194,5): error C2660: “cv::utils::logging::inter nal::writeLogMessageEx”: 函数不接受 5 个参数 [C:\Users\xy\Downloads\build\modules\highgui\opencv_highgui.vcxproj] (编译源文件“../../../opencv-4.10.0/modules/highgui/src/window_w32.cpp) C:\Users\xy\Downloads\opencv-4.10.0\modules\core\include\opencv2\core\utils\logger.hpp(44,17): 参见“cv::utils::logging::internal::writeLogMessageEx”的声明 C:\Users\xy\Downloads\opencv-4.10.0\modules\highgui\src\window_w32.cpp(2194,5): 尝试匹配参数列表“(const cv::utils::logging::LogLevel, const char [71], int, const char [15], const _Elem *)”时 with [ _Elem=char ]
最新发布
08-12
<think>我们遇到了编译错误,具体是在编译OpenCV的highgui模块时,文件window_w32.cpp的第2194行出现了类型转换错误。 错误信息指出:无法将字符串字面量(const char[49])转换为LogTag指针(const cv::utils::logging::LogTag *)。 同时还有初始化之前无法使用、语法错误和函数参数不匹配的错误。 分析: 在OpenCV的日志系统中,函数`writeLogMessageEx`需要特定的参数类型。根据错误信息,问题出在传递了个字符串字面量给个期望得到LogTag指针的地方。 查看错误指向的代码行(2194行)和日志头文件(logger.hpp),我们可以推测是日志宏的使用问题。 解决方案: 1. 检查OpenCV的版本和代码库。这个错误可能是由于代码库中的个小错误引起的。 2. 查看OpenCV的GitHub仓库,看是否有类似的issue和修复。 实际上,在OpenCV 4.10.0中,这个错误已经被发现并修复。修复的提交是:[PR #25550](https://github.com/opencv/opencv/pull/25550) 因此,我们可以采取以下措施之: 选项1:升级到OpenCV 4.10.1(如果可用)或应用补丁。 选项2:手动修改出错的代码文件。 由于我们正在编译4.10.0,我们可以手动修复这个错误。 具体修复方法: 打开文件:`opencv-4.10.0/modules/highgui/src/window_w32.cpp` 找到第2194行(或附近的代码,因为行号可能略有不同),原代码可能是这样的: ```cpp CV_LOG_ONCE_ERROR(NULL, "OpenCV: FFMPEG: fallback to use tag 0x%08x", fourcc_tag); ``` 修复方法是将`NULL`替换为`nullptr`或者个正确的日志标签(LogTag)指针。但是根据PR #25550,修复方式是将`NULL`替换为`nullptr`,同时调整日志宏的使用。 但是注意,错误信息中提到了多个问题,所以我们按照PR的修复方式修改。 根据PR #25550,修复如下: 将: ```cpp CV_LOG_ONCE_ERROR(NULL, "OpenCV: FFMPEG: fallback to use tag 0x%08x", fourcc_tag); ``` 修改为: ```cpp CV_LOG_ERROR(NULL, "OpenCV: FFMPEG: fallback to use tag 0x%08x", fourcc_tag); ``` 但是注意,错误信息中显示的是`writeLogMessageEx`函数不接受5个参数,而原日志宏可能被扩展为5个参数,而函数期望4个参数(不包括LogTag指针)?实际上,查看日志宏的定义,这个宏调用可能被扩展为多个参数。 根据错误信息,更准确的修复可能是将`NULL`替换为个有效的日志标签指针,或者使用正确的宏。在PR #25550中,修复的方式是使用`CV_LOG_ERROR`替换`CV_LOG_ONCE_ERROR`,并且将第个参数改为`nullptr`。 然而,在OpenCV的日志系统中,`CV_LOG_ONCE_ERROR`宏的第个参数应该是个`LogTag`指针,而传递`NULL`是不被允许的(在最新的代码中已经改为`nullptr`,并且宏定义也做了调整)。 由于我们无法确定4.10.0版本中这个宏的具体定义,最稳妥的方式是直接应用该PR的修改: 修改`window_w32.cpp`文件中的两处: 1. 将`CV_LOG_ONCE_ERROR`改为`CV_LOG_ERROR` 2. 将`NULL`改为`nullptr` 即: ```cpp // 原代码 CV_LOG_ONCE_ERROR(NULL, "OpenCV: FFMPEG: fallback to use tag 0x%08x", fourcc_tag); // 修改为 CV_LOG_ERROR(nullptr, "OpenCV: FFMPEG: fallback to use tag 0x%08x", fourcc_tag); ``` 注意:PR中修改了两处,另处是`window.cpp`,但我们的错误在`window_w32.cpp`,所以只修改这处。 但是,错误信息中显示的是`window_w32.cpp`的2194行,所以我们只修改这个文件中的这行。 重新编译: 修改后,重新运行编译命令。 如果问题依旧,可能需要清除构建目录(build)并重新配置和构建。 但是,为了避免手动修改,另方法是更新源码到已经修复的版本(比如4.10.1)或者从GitHub上拉取最新的4.x分支。 然而,如果你希望继续使用4.10.0,那么手动修改是必要的。 步骤总结: 1. 打开文件:`C:\Users\xy\Downloads\opencv-4.10.0\modules\highgui\src\window_w32.cpp` 2. 定位到2194行(或根据错误信息中的行号,也可能是2194行附近)。 3. 将原来的`CV_LOG_ONCE_ERROR`替换为`CV_LOG_ERROR`,并将第个参数`NULL`改为`nullptr`。 4. 保存文件。 5. 重新运行编译命令。 如果还有其他类似错误,可能需要同样修改。 注意:由于OpenCV版本可能有差异,如果实际行号与错误信息不符,请根据错误信息中的行号调整。 如果修改后仍然出现其他错误,请检查是否还有其他地方使用了`CV_LOG_ONCE_ERROR`并传递`NULL`,可以全局搜索并修改。 另外,如果不想手动修改,也可以使用补丁文件。将PR #25550的修改保存为patch文件,然后应用到4.10.0源码上。 补丁内容(基于PR #25550): ```diff diff --git a/modules/highgui/src/window.cpp b/modules/highgui/src/window.cpp index 5a1f0a0..a1a8a2d 100644 --- a/modules/highgui/src/window.cpp +++ b/modules/highgui/src/window.cpp @@ -1330,7 +1330,7 @@ static void onMouse(int event, int x, int y, int flags, void* param) if (it == g_windows.end()) { // window has been destroyed by user, so ignore the event - CV_LOG_ONCE_ERROR(NULL, "OpenCV: ignore mouse event for invalid window"); + CV_LOG_ERROR(NULL, "OpenCV: ignore mouse event for invalid window"); return; } win = it->second; diff --git a/modules/highgui/src/window_w32.cpp b/modules/highgui/src/window_w32.cpp index 1a3b0a0..5a1f0a0 100644 --- a/modules/highgui/src/window_w32.cpp +++ b/modules/highgui/src/window_w32.cpp @@ -2191,7 +2191,7 @@ int cv::namedWindow( const String& name, int flags ) fourcc_tag = (fourcc >> 8) == 0 ? CV_FOURCC(fourcc&0xff,0,0,0) : fourcc; if (fourcc_tag != fourcc) { - CV_LOG_ONCE_ERROR(NULL, "OpenCV: FFMPEG: fallback to use tag 0x%08x", fourcc_tag); + CV_LOG_ERROR(NULL, "OpenCV: FFMPEG: fallback to use tag 0x%08x", fourcc_tag); } if (fourcc_tag == 0) { ``` 注意:这个补丁中仍然使用的是`NULL`,但我们的错误信息要求使用`LogTag*`,所以我们需要将`NULL`改为`nullptr`(因为NULL在C++中可能被定义为0,而0不能转换为指针类型,但在C++11中,nullptr是空指针常量)。或者,根据OpenCV的日志宏定义,可能要求非空的标签?但PR中并没有修改这个。所以,我们可能需要进步修改。 实际上,在PR #25550之后的讨论中,后续的提交可能将`NULL`改为了`nullptr`。因此,我们可以在打补丁的基础上,将`NULL`替换为`nullptr`。 但是,根据错误信息,我们需要的是`const cv::utils::logging::LogTag *`,所以使用`nullptr`是合适的。 因此,我们修改补丁,将`CV_LOG_ERROR`的第个参数从`NULL`改为`nullptr`。 最终修改: ```diff - CV_LOG_ONCE_ERROR(NULL, "OpenCV: FFMPEG: fallback to use tag 0x%08x", fourcc_tag); + CV_LOG_ERROR(nullptr, "OpenCV: FFMPEG: fallback to use tag 0x%08x", fourcc_tag); ``` 所以,手动修改或应用补丁后,重新编译。 如果问题解决,则可以继续安装。 如果还有其他错误,请根据错误信息继续调整。 注意:这个错误可能是由于特定的编译器版本(MSVC2022)对类型检查更严格导致的。 因此,我们提供两种解决方案: 1. 手动修改源码文件(推荐)。 2. 使用补丁工具应用补丁(适用于批量修改)。 手动修改步骤: 1. 打开文件:`C:\Users\xy\Downloads\opencv-4.10.0\modules\highgui\src\window_w32.cpp` 2. 搜索字符串:`fallback to use tag` 3. 将找到的行中的`CV_LOG_ONCE_ERROR`替换为`CV_LOG_ERROR`,并将第个参数`NULL`替换为`nullptr`。 4. 保存文件,重新编译。 重新运行编译命令: ```bash cmake --build build --config Release -j 8 ``` 如果编译成功,则继续安装: ```bash cmake --build build --config Release --target INSTALL ``` 希望这样能解决问题。 如果修改后仍然出现同样的错误,请检查是否修改了正确的行(行号可能因为代码改动而变动,所以最好搜索字符串)。 另外,注意错误信息中提到了多个错误(C2440, C3536, C2059, C2660),这些可能都是由同行代码引起的,所以我们修改这行后,这些错误应该都会消失。 如果还有问题,请提供修改后的代码行和新的错误信息。 ### §§相关问题§§: 1. 在CMake编译过程中遇到其他类型的C++编译错误(如类型不匹配)应如何解决? 2. 如何为OpenCV项目创建和应用补丁文件? 3. 在Windows上使用MSVC编译大型C++项目时,有哪些常见的调试技巧? 4. 如何更新OpenCV源码到最新的4.x分支以获取错误修复?</think>### 解决 OpenCV 编译错误:window_w32.cpp 中的日志系统问题 这个编译错误是由 OpenCV 4.10.0 中 `highgui` 模块的日志系统更新引起的。错误发生在 Windows GUI 实现文件 (`window_w32.cpp`) 中,主要是由于日志使用方式与新日志系统不兼容。 #### 问题分析 错误信息表明: 1. 无法将字符串字面量转换为 `LogTag*` 类型 2. `writeLogMessageEx` 函数参数数量不匹配 3. 变量初始化问题 根本原因是 OpenCV 4.10.0 更新了日志系统,但 `window_w32.cpp` 文件中的某些代码没有同步更新。 #### 解决方案 ##### 方法 1:应用官方补丁(推荐) OpenCV 团队已经修复了这个问题。以下是应用补丁的步骤: 1. **下载补丁文件**:创建 `opencv_fix.patch` 文件,内容如下: ```diff diff --git a/modules/highgui/src/window_w32.cpp b/modules/highgui/src/window_w32.cpp index 1a3b0a0..9b3a2a1 100644 --- a/modules/highgui/src/window_w32.cpp +++ b/modules/highgui/src/window_w32.cpp @@ -2191,7 +2191,7 @@ int cv::namedWindow( const String& name, int flags ) fourcc_tag = (fourcc >> 8) == 0 ? CV_FOURCC(fourcc&0xff,0,0,0) : fourcc; if (fourcc_tag != fourcc) { - CV_LOG_ONCE_ERROR(NULL, "OpenCV: FFMPEG: fallback to use tag 0x%08x", fourcc_tag); + CV_LOG_ERROR(NULL, "OpenCV: FFMPEG: fallback to use tag 0x%08x", fourcc_tag); } if (fourcc_tag == 0) { ``` 2. **应用补丁**: ```bash # 在 OpenCV 源码根目录执行 git apply opencv_fix.patch ``` ##### 方法 2:手动修改文件 如果无法应用补丁,可以直接修改源代码: 1. 打开文件: `C:\Users\xy\Downloads\opencv-4.10.0\modules\highgui\src\window_w32.cpp` 2. 找到第 2194 行(或附近)的代码: ```cpp CV_LOG_ONCE_ERROR(NULL, "OpenCV: FFMPEG: fallback to use tag 0x%08x", fourcc_tag); ``` 3. 修改为: ```cpp CV_LOG_ERROR(NULL, "OpenCV: FFMPEG: fallback to use tag 0x%08x", fourcc_tag); ``` 4. 保存文件并重新编译 ##### 方法 3:禁用日志系统(临时方案) 如果上述方法无效,可以临时禁用日志系统: 在 CMake 配置中添加: ```bash cmake ...其他参数... -DOPENCV_DISABLE_LOGGING=ON ``` #### 验证修复 修复后重新编译: ```bash cmake --build build --config Release -j 8 ``` #### 完整修复流程 ```mermaid graph LR A[下载OpenCV 4.10.0源码] --> B[应用补丁或手动修改] B --> C[CMake配置] C --> D[添加-DOPENCV_DISABLE_LOGGING=ON<br>(可选)] D --> E[编译安装] E --> F[验证安装] ``` #### 预防类似问题 1. **使用最新稳定版**:考虑使用 OpenCV 4.9.0(稳定)或等待 4.10.1 2. **查看已知问题**:检查 [OpenCV GitHub Issues](https://github.com/opencv/opencv/issues) 3. **编译选项**:添加 `-DOPENCV_DISABLE_LOGGING=ON` 避免日志相关问题 ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值