linux中由于编译选项-D_OS64BIT导致的核心已转储问题排查解决:
原因:
a.so b.so a.so使用b.so
程序1 程序2 使用a.so
程序1运行正常,程序2启动后提示核心已转储。
程序1和程序2运行的代码都一致,只执行创建xApplication app;
gdb调试程序2,bt(backtrace)查看崩溃堆栈,异常出现在b.so中,创建xApplication时会创建b.so中的对象。
解决方法:
应该是Makefile编译选项导致的,对比两个程序的Makefile;
程序1的Makefile是通过qmake -project生成.pro,然后qmake生成Makefile,编译出来的程序正常执行;
程序2有Makefile,对比Makefile,发现程序2的Makefile编译选项中有**-D_OS64BIT**;
去掉程序2Makefile中的-D_OS64BIT,重新编译后可以正常运行了。
-D_OS64BIT是一个预处理器宏定义,通常用于在代码中区分32位和64位操作系统环境。
基本用法:
1.在编译命令中定义
bash/ g++ -D_OS64BIT xxx.cpp -o xxx.o
2.在Qt项目(.pro文件)中定义
qmake/
#添加全局定义
DEFINES += _OS64BIT#或者有条件地定义
contains(QT_ARCH, x86_64) {DEFINES += _OS64BIT
}
实际应用场景
1.平台相关代码处理
cpp
#ifdef _OS64BIT// 64位特定代码using MyInt = int64_t;
#else// 32位代码using MyInt = int32_t;
#endif
2.内存处理差异
cpp
void* allocateMemory(size_t size){
#ifdef _OS64BIT// 64位操作系统可以使用更大内存return malloc(size*2);
#elsereturn malloc(size);
#endif
}
更好的替代方案
虽然*OS64BIT可用,但更推荐使用标准化的检测方法:
1.使用编译器预定义宏
cpp
#if defined(__x86_64__) || defined(__ppc64__) || defined(__arch64__)// 64位系统
#else// 32位系统
#endif
2.使用Qt内置检测
cpp
#include <QtGlobal>
#if QT_POINTER_SIZE = 8// 64位系统
#else// 32位系统
#endif
常见问题解决
1.宏未生效:
确保编译命令正确定义了宏
检查是否有其他地方取消了定义
(#undef OS64BIT)
2.与其它宏冲突:
考虑使用更独特的宏名,如MYAPP_OS64BIT
3.跨平台兼容性:
在Windows上可能需要使用WIN64代替。
最佳实践
1.在头文件中集中管理平台相关定义
cpp
// platform_config.h
#if !defined(IS_64BIT)#if defined(__x86_64__) || defined(__ppc64__) || defined(__aarch64__) || defined(_M_X64)#define IS_64BIT 1#else#define IS_64BIT 0#endif
#endif
2.使用静态断言验证:
cpp
static_assert(sizeof(void*)==(IS_64BIT ? 8 : 4), "Pointer sieze mismatch");
3.在构建系统中明确定义:
cmake
#CMake示例
if(CMAKE_SIZEOF_VOID_P EQUAL 8)add_compile_definitions(IS_64BIT=1)
else()add_compile_definitions(IS_64BIT=0)
endif()