首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >QLibrary -导入类

QLibrary -导入类
EN

Stack Overflow用户
提问于 2014-10-07 10:51:36
回答 1查看 6.6K关注 0票数 6

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

现在,由于我希望这样做,即使在修改库时,也不需要再次编译另一个项目,所以我开始使用QLibrary。

但是..。我不能导入一个类。或者更好的是,我可以导入类,但是不能访问它的方法。

这就是我做的例子。

这是类声明:

代码语言:javascript
复制
class TESTDLL_LIBSHARED_EXPORT TestDLL_lib
{

public:
    TestDLL_lib();

    int a;
    int b;
    int c;

    int getValues();
}; 

这就是实施:

代码语言:javascript
复制
#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();
}

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

代码语言:javascript
复制
#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(),因为

代码语言:javascript
复制
main.obj:-1: error: LNK2001: unresolved external symbol "__declspec(dllimport) public: int __thiscall TestDLL_lib::getValues(void)" (__imp_?getValues@TestDLL_lib@@QAEHXZ)

我怎么才能解决这个问题?可以从导入的类调用方法吗?

谢谢。。

EN

回答 1

Stack Overflow用户

回答已采纳

发布于 2014-10-07 12:24:22

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

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

不要忘记使您的d‘dtor virtual和添加一个virtual dtor到接口。如果不这样做,则无法通过接口指针安全地使用delete实例。

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

代码语言:javascript
复制
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();
}
票数 8
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/26234327

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档