6

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();
    }
}

これで、オブジェクトのすべてのデータ値 ( ab、 ) にアクセスできます(さらに、DLL でそれらを変更すると、再構築せずにプログラムでも変更されます) 。cmyClassInstancemyClassInstance->getValues()

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

どうすればこれを解決できますか? インポートされたクラスからメソッドを呼び出すことは可能ですか?

ありがとうございました..

4

1 に答える 1

8

実行時にインポートされたクラスのメソッドを呼び出すことはできません。これは、コンパイラがこれらの呼び出しを実行時ではなくコンパイル時にリンクするためです (これは実行できません)。私たちの古き良き友人である vtable によって解決策が提供されます。

インターフェイスを実装するクラスのメソッドを呼び出すことができvirtualます (インターフェイスは実行時に「インポート」されません)。virtualこれは、 (おそらく純粋仮想) メソッドを使用してインターフェイスを定義するクラスを定義することを意味します。TestDLL_lib次に、そのインターフェースを継承し、メソッドを実装します。そのインターフェイスを介してインスタンスを参照し、そのインターフェイスを介してメソッドを呼び出し、 s vtableTestDLL_libによって「置き換えられた」インターフェイスの vtable を介してそれらを効果的に呼び出します。TestDLL_lib

d'tor を作成し、インターフェイスに dtor をvirtual追加することを忘れないでください。そうしないと、インターフェイス ポインターを介してvirtual安全にインスタンス化できません。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();
}
于 2014-10-07T12:24:22.237 に答える