gpt4 book ai didi

c++ - QLibrary - 导入一个类

转载 作者:塔克拉玛干 更新时间:2023-11-03 00:24:20 25 4
gpt4 key购买 nike

我有一个 QT 库,我想将它导入到另一个项目中。

现在,既然我想要那个,即使我修改了库,另一个项目也不需要重新编译,我开始使用 QLibrary。

但是...我无法导入类。或者更好的是,我可以导入该类,但我无法访问它的方法。

这是我做的例子。

这是类声明:

class TESTDLL_LIBSHARED_EXPORT TestDLL_lib
{

public:
TestDLL_lib();

int a;
int b;
int c;

int getValues();
};

这是实现:

#include "testdll_lib.h"

TestDLL_lib::TestDLL_lib()
{
a = 10;
b = 20;
c = 30;
}

int TestDLL_lib::getValues()
{
return a+b+c;
}

extern "C" TESTDLL_LIBSHARED_EXPORT TestDLL_lib* create_TestDLL_lib()
{
return new TestDLL_lib();
}

虽然这是主文件,但在另一个项目中:

#include <testdll_lib.h>
#include <QDebug>
#include <QLibrary>

int main(int argc, char *argv[])
{
QLibrary library("TestDLL_lib");
if (library.load())
{
typedef TestDLL_lib* (*create_TestDLL_lib_fun)();
create_TestDLL_lib_fun create_TestDLL_lib = (create_TestDLL_lib_fun)library.resolve("create_TestDLL_lib");

if (create_TestDLL_lib)
{
TestDLL_lib *myClassInstance = create_TestDLL_lib();

if (myClassInstance)
{
//qDebug() << QString::number(myClassInstance->getValues());
qDebug() << QString::number(myClassInstance->a) + " " + QString::number(myClassInstance->b) + " " + QString::number(myClassInstance->c);
}
}

library.unload();
}
}

现在,我可以访问对象 myClassInstance 的所有数据值(abc) > (而且,如果我在 DLL 中更改它们,它们也会在程序中更改而无需重建)但我无法调用 myClassInstance->getValues() 因为我得到

main.obj:-1: error: LNK2001: unresolved external symbol "__declspec(dllimport) public: int __thiscall TestDLL_lib::getValues(void)" (__imp_?getValues@TestDLL_lib@@QAEHXZ)

我该如何解决这个问题?是否可以从导入的类中调用方法?

谢谢你..

最佳答案

您不能在运行时导入的类上调用方法。这是因为编译器在编译时而不是在运行时链接这些调用(它不能)。我们的好 friend vtable 提供了一条出路:

您可以在实现接口(interface)的类上调用 virtual 方法(该接口(interface)在运行时不“导入”)。这意味着使用 virtual(可能是纯虚拟)方法来定义定义接口(interface)的类。 TestDLL_lib 将继承该接口(interface),实现这些方法。您可以通过该接口(interface)引用 TestDLL_lib 实例并通过该接口(interface)调用方法,有效地通过接口(interface)的 vtable 调用它们,该接口(interface)被 TestDLL_lib 的 vtable“取代” .

不要忘记让你的d'tor virtual 并在界面上添加一个virtual dtor。如果不这样做,则无法通过接口(interface)指针安全地删除实例。

我可能还会解释为什么您可以访问成员,但不能调用“导入”类上的函数。成员按内存位置访问,内存位置由编译器单独定义。因此,编译器生成代码来访问成员,而无需引用任何类的符号(方法等)。这反过来导致没有链接依赖性。但是请注意,如果更改类,例如,您将需要重新编译 DLL 和使用 DLL 的应用程序。添加或删除成员,因为这会改变内存布局。

class TestInterface
{
public:
virtual ~TestInterface()
{
}

virtual int getValues() = 0;
}

class TESTDLL_LIBSHARED_EXPORT TestDLL_lib : public TestInterface
{

public:
TestDLL_lib();
virtual ~TestDLL_lib();

int a;
int b;
int c;

int getValues() override; // MSVC may not support "override"
};

// return pointer to interface!
// TestDLL_lib can and should be completely hidden from the application
extern "C" TESTDLL_LIBSHARED_EXPORT TestInterface *create_TestDLL_lib()
{
return new TestDLL_lib();
}

关于c++ - QLibrary - 导入一个类,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26234327/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com