2

我正在开发一个使用 Qt (gui lib)、VTK (graphics lib) 和另一个非常晦涩的库的 C++ 项目,我不会提及它的名称,而是将其称为 LIB_X。该项目使用 Qt 作为 gui 组件和 VTK(更准确地说是 VTK 提供的支持 Qt 的 QVTKWidget 扩展)用于渲染几何图形。它使用 LIB_X 来收集和操作几何图形。

问题是事实证明 LIB_X 实际上使用 VTK(我不知道它在哪里以及如何使用,它是封闭源代码)。起初没有问题,用两个链接的库编译都很好,但在某些时候我调用了某个(并且非常需要的)LIB_X 函数,编译导致一堆关于VTK lib / obj的东西已经定义了在 LIB_X dll 的错误中。

例如(注意这是 /FORCE:MULTIPLE 所以这里是一个警告,如果你想要没有 /FORCE:MULTIPLE 的错误,请告诉我,我会发布它):

1>LIB_X.lib(LIB_X.dll) : warning LNK4006: "public: __thiscall std::vector<double,class std::allocator<double> >::~vector<double,class std::allocator<double> >(void)" (??1?$vector@NV?$allocator@N@std@@@std@@QAE@XZ) already defined in vtkCommon.lib(vtkInformationDoubleVectorKey.obj);

我尝试使用 /FORCE:MULTIPLE,起初这似乎是一个奇迹,但我在代码中遇到随机错误,这些错误主要会导致堆错误。我决定从主项目中删除所有对 LIB_X 的引用,并创建一个静态库来处理所有 LIB_X 内容。我不是 C++ 专家,所以我不确定当您使用预编译库时它如何处理库冲突,但是在将静态库链接到我的主项目时,我仍然收到库冲突错误,所以我仍然必须使用 /FORCE:MULTIPLE。

一旦我有了静态库,似乎随机错误就消失了,我可以通过静态库在主项目中使用 LIB_X 方法做很多事情,但是不知从何而来,我在我的主项目中添加了一个新数据成员类(双精度的 std::vector),突然我在我的静态库的一个方法中遇到了堆错误。如果我注释掉新的数据成员,静态库的方法会运行良好。我讨厌给出当前的错误,因为老实说,我不确定检查它是否值得,但无论如何它都在这里,以防万一它可以提供帮助:

注意:它在大约第 151 行崩溃到 xutility,弹出断言:“文件:dbgheap.c 行:1279 表达式:_CrtIsValidHeapPointer(pUserData)”

将向量向量双精度添加到向量向量向量双精度后出现错误,在 push_back 行上崩溃:

std::vector<std::vector<double>> tmpVec;
for(srvl_iter = srvl.begin(); srvl_iter != srvl.end(); ++srvl_iter)
{
 tmpVec.push_back((*srvl_iter).getControlPoints());
}
this->_splines.push_back(tmpVec); //CRASH

当我向我的主项目添加一个新的数据成员(与静态库分开!)时,它才开始在这里崩溃。注释掉新的数据成员可以消除错误。

std::vector<std::vector<std::vector<double>>> _geometry; 

所以, /FORCE:MULTIPLE 似乎很糟糕,我得到了对我来说没有意义的随机错误。还有其他解决方案吗?我搞砸了吗?我可以用 LIB_X 的 VTK 链接做些什么吗?

4

2 回答 2

1

LNK4006在将我的应用程序链接到大量使用 的库(称为库 LIB_Y)时,我遇到了一堆错误std::vector<std::string>,我在我的应用程序中也这样做了。经过一番试验,我找到了一个可行的解决方案——将 LIB_Y 包装在一个单独的 DLL 中,该 DLL 调用 LIB_Y(比如 LIB_Y_WRAPPER),然后将主应用程序链接到 LIB_Y_WRAPPER。

要尝试我的建议,您需要:

  1. 将“处理所有 LIB_X 内容的静态库”从静态 LIB 项目更改为 DLL 项目(我将其称为 LIB_X_WRAPPER)。
  2. 确保 LIB_X_WRAPPER 的头文件不包含任何 LIB_X 头文件。这一点非常重要,因为包装器需要将您的应用程序与 LIB_X 头文件中声明的数据类型(例如std::vector<double>)完全隔离开来。仅从 LIB_X_WRAPPER 的源文件中引用 LIB_X 的头文件。
  3. 更改静态库中所有类和函数的声明,以确保它们从 DLL 导出(如果您需要有关从 DLL 导出的详细信息,请参阅此答案)。

这个解决方案对我有用,因为它使 LIBY 使用的类的实例化(编译器生成的函数)与我的应用程序中std::vector<std::string>的实例化完全分开。std::vector<std::string>

顺便说一句,我怀疑您看到的崩溃的原因(您评论它在 的析构函数中std::vector<double>)是因为 std::vector<double>您的应用程序中的实例化与 LIB_X 中的不同。

于 2010-01-10T13:56:26.223 回答
0

注释掉可能只是随机运气 - 如果您正在破坏堆,您并不总是会立即看到它,但是 stl 向量会左右分配和释放东西,所以难怪它会发现错误。

一些库要求您按特定顺序包含内容。我不知道为什么,因为对我来说似乎放弃并说你不能正确设计图书馆,但这是可悲的事实。不过,只要您不在包含 vtk 的任何地方包含此 lib_x 就可以了。

但是,他们可能正在争夺某些资源或使用不当的东西,这使他们无法一起工作。如果您能够通过隔离它们来使编译正常工作,但它仍然失败,那么是的,您只是不走运,因为这只是这个 lib_x 设计方式的失败,而且因为它太晦涩难懂,所以不太可能是针对所有用途进行了彻底调试。当某些东西没有被广泛使用时,它通常最终成为可以在开发人员的机器和项目上运行的东西,但不一定是其他任何人的东西。

于 2010-01-09T21:19:48.027 回答