QLibrary - 导入一个类

fra*_*i87 5 c++ qt qlibrary qt5

我有一个QT库,我想在另一个项目中导入它.

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

但是......我无法导入课程.或者更好,我可以导入该类,但我无法访问其方法.

这是我做的例子.

这是类声明:

class TESTDLL_LIBSHARED_EXPORT TestDLL_lib
{

public:
    TestDLL_lib();

    int a;
    int b;
    int c;

    int getValues();
}; 
Run Code Online (Sandbox Code Playgroud)

这个实施:

#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();
}
Run Code Online (Sandbox Code Playgroud)

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

#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();
    }
}
Run Code Online (Sandbox Code Playgroud)

现在,我可以访问所有的数据值(a,b,c)的对象myClassInstance(如果我在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)
Run Code Online (Sandbox Code Playgroud)

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

谢谢..

dom*_*om0 6

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

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

不要忘记制作你virtualvirtual设备并在界面上添加一个dtor.如果不这样做,则无法delete通过接口指针安全地实例化.

我也可以解释为什么你可以访问成员,但不能在"导入"类上调用函数.成员由内存位置访问,内存位置仅由编译器定义.因此,编译器生成代码以访问成员,而无需引用任何类的符号(方法等).这反过来导致没有链接依赖性.但是请注意,如果更改类,则需要使用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();
}
Run Code Online (Sandbox Code Playgroud)