0

我可以在 C++ DLL 中使用 C++SetDllDirectoryLoadLibrary命令来加载另一个 DLL 吗?我试过像这样使用它们:

可执行文件调用第一个 DLL,然后第一个 DLL 加载第二个 DLL,然后第二个 DLL 进行计算...

但是当我运行可执行文件时,我收到以下错误消息:

此应用程序已请求运行时以不寻常的方式终止它。请联系应用程序支持团队了解更多信息。

直接链接到可执行文件时,第二个 DLL 工作正常!

这是我的可执行文件中的代码:

#include <windows.h>
#include <iostream>

int main(){
    HINSTANCE hDLL_Link=NULL;
    SetDllDirectory((LPCWSTR)L"C:\\Users\\MC\\Documents\\2014_07_01a_FDD_VC2008\\test_call_DLL\\EXE_calls_Link_DLL\\Release");
    hDLL_Link=LoadLibrary((LPCWSTR)L"Link_DLL.dll");
    if(hDLL_Link==NULL) std::cout<<"did not load"<<'\n';
    typedef void (*Ptr_OPS_Link)();
    Ptr_OPS_Link Ptr_OPS_Link_0;
    Ptr_OPS_Link_0=(Ptr_OPS_Link)GetProcAddress(hDLL_Link,"OPS_Link");
    Ptr_OPS_Link_0();
    FreeLibrary(hDLL_Link);
    system("pause");
}

这是第一个 DLL 中的代码:

#include "Link.h"

extern "C" __declspec(dllexport)
void OPS_Link(){
    Link*Link_Ptr_Object=NULL;
    if(Link_Ptr_Object==NULL){
        Link_Ptr_Object=new Link();
    }
    if(Link_Ptr_Object==NULL){
        //can not throw inside __declspec(dllexport) functions marked extern "C" that's why std::cout is implemented:
        std::cout<<"Error: could not link to FDD DLL"<<'\n';
        system("pause");
    }
    delete Link_Ptr_Object;
    Link_Ptr_Object=NULL;
}

Link::Link()
:m_void_Ptr_ObjectByDLL(NULL){
    HINSTANCE hDLL=NULL;//handle to DLL
    SetDllDirectory((LPCWSTR)L"C:\\Software\\Octave-3.6.1\\bin\\");
    hDLL=LoadLibrary((LPCWSTR)L"C:\\Users\\MC\\Documents\\2014_07_01a_FDD_VC2008\\Executable\\Release\\FDD_DLL.dll");
    if(hDLL==NULL){
        throw "DLL loading could not be done";
    }else if(hDLL!=NULL){
        typedef void (*Ptr_OPS_FDD)(std::string, int, int);
        Ptr_OPS_FDD Ptr_OPS_FDD_0;//pointer to procedure inside DLL
        Ptr_OPS_FDD_0=NULL;
        Ptr_OPS_FDD_0=(Ptr_OPS_FDD)GetProcAddress(hDLL,"OPS_FDD");
        if(Ptr_OPS_FDD_0==NULL){
            FreeLibrary(hDLL);
            throw "DLL exported function address could not be determined";
        }else{
            //run the procedure inside DLL:
            Ptr_OPS_FDD_0("FDD_INPUT_Truss_Bridge_Data2_Ambient_Inch_11Channels_77824Samples_SamplingFreq_256Hz.txt",11,256);//LabScaleTruss
            //Ptr_OPS_FDD_0("FDD_INPUT_Ambient_EW_15Channels_3000Samples_SamplingFreq_20Hz.txt",15,20);//AmbientEW
            //Ptr_OPS_FDD_0("FDD_INPUT_Meriden_3Channels(3_5_8)_3686400Samples_SamplingFreq_2048Hz.txt",3,2048);//MeridenBridge
            FreeLibrary(hDLL);
        }
    }
}
4

3 回答 3

2

您的代码中有几件事可能会导致失败:

  1. 如果无法加载 DLL,您不会退出:
  2. 您正在传递内部使用动态分配的对象,因此将使用堆管理器。

对于1.上述情况,您的 main() 函数仅cout在找不到库时才执行简单操作。然而,main函数并没有退出,而是像找到了库一样继续执行。

对于2.上述情况,std::string作为参数传递给 DLL 函数很容易出错并且不推荐,除非您确切知道自己在做什么。容易出错的原因是

  • 包含函数调用的 DLL 可能是使用与调用函数的 DLL 不同的选项集构建的。这些不同的选项可能会导致std::string实现方式、内存布局等方面 的差异。

  • 包含函数调用的 DLL 可能是由与调用函数的 DLL 不同的编译器版本构建的。同样,相同的问题与不同的实现std::string

  • 使用 std::string 的 DLL 和模块可能不是使用DLL versionC 运行时库构建的。如果未使用运行时库的 DLL 版本构建和链接 DLL/模块,则 DLL 将使用与模块不同的堆。由于使用了不同的内存堆,对 std::string 的任何操作都将无效。

所以简而言之,除非你能保证

  1. 您正在使用完全相同版本的编译器和编译器选项构建模块和 DLL。
  2. 您将所有模块链接到运行时库的 DLL 版本。

然后std::string作为参数传递,一般来说,传递任何维护动态分配内存的对象,都可能或将导致运行时错误。

于 2014-07-28T16:45:14.577 回答
1

除了不充分的错误处理和跨模块边界使用标准库之外,还有两件事需要考虑。

我可以在 dll 中使用 SetDllDirectory 来...吗?

是的,你可以,但你不应该!(等待发生的错误)。

为什么 ?因为唯一负责改变环境的实体是主应用程序。库代码(静态或 dll)不知道它将在哪个应用程序中使用。它可能在某些程序中正常工作,而在其他程序中可能会失败。

我可以在 dll 中使用 C++ LoadLibrary/FreeLibrary 来...吗?

是的,你可以,但不要在 dllmain 函数中使用它们,因为它会死锁你的程序。

于 2014-07-28T17:03:21.280 回答
0

我解决了这个问题,并在这里展示了如何:

我更改了可执行文件和第一个 DLL 中的代码,如下所示,以考虑错误处理,并且我添加了“ return 0;”,现在可执行文件链接到第一个 DLL,它工作完美......实际上问题是main需要return一些东西......我在 DLL 边界处std::string用“ ”替换所有“ ” char*...顺便说一下,我想开发两个 DLL 而我SetDllDirectory在第一个 DLL 中使用“”的原因是我想用 C# 调用一个 DLL GUI,问题是C#中没有“ SetDllDirectory”命令可用,因此,我想出了开发两个DLL的想法,在第一个DLL中,我将使用“SetDllDirectory" 来处理所需的依赖项(DLL 依赖于 Octave 和 Octave Bin 目录),然后我开发了第二个 DLL 来执行实际计算......我知道有一些像 " [DllImport("Kernel32.dll")]" 这样的方法,从那里我们可以在 C# 中使用“ SetDllDirectory”,但这种方法看起来很痛苦。

可执行文件内的更正代码:

    #include <windows.h>
#include <iostream>

int main(){
    try{
        HINSTANCE hDLL_Link=NULL;
        hDLL_Link=LoadLibrary((LPCWSTR)L"Link_DLL.dll");
        if(hDLL_Link==NULL){
            throw "Link DLL did not load";
        }else{
            typedef void (*Ptr_OPS_Link)();
            Ptr_OPS_Link Ptr_OPS_Link_0=NULL;
            Ptr_OPS_Link_0=(Ptr_OPS_Link)GetProcAddress(hDLL_Link,"OPS_Link");
            if(Ptr_OPS_Link_0==NULL){
                throw "Link DLL exported function not found";
                FreeLibrary(hDLL_Link);
            }else{
                Ptr_OPS_Link_0();
                FreeLibrary(hDLL_Link);
            }
        }
    }
    catch(char*char_Ptr_Exception){
        std::cerr<<"Error: "<<char_Ptr_Exception<<'\n';
    }
    system("pause");
    return 0;
}

第一个 DLL 中的更正代码:

    #include "Link.h"

extern "C" __declspec(dllexport)
void OPS_Link(){
    Link*Link_Ptr_Object=NULL;
    if(Link_Ptr_Object==NULL){
        Link_Ptr_Object=new Link();
    }
    if(Link_Ptr_Object==NULL){
        ////can not throw inside __declspec(dllexport) functions marked extern "C" that's why std::cout is implemented:
        //std::cout<<"Error: could not link to FDD DLL"<<'\n';
        system("pause");
    }
    delete Link_Ptr_Object;
    Link_Ptr_Object=NULL;
}

Link::Link()
:m_void_Ptr_ObjectByDLL(NULL){
    HINSTANCE hDLL=NULL;//handle to DLL
    SetDllDirectory((LPCWSTR)L"C:\\Software\\Octave-3.6.1\\bin\\");
    //path relative to executable (C# executable or C++ executable)
    hDLL=LoadLibrary((LPCWSTR)L"FDD_DLL.dll");
    if(hDLL==NULL){
        throw "FDD DLL did not load";
    }else if(hDLL!=NULL){
        typedef void (*Ptr_OPS_FDD)(char*, int, int);
        Ptr_OPS_FDD Ptr_OPS_FDD_0;//pointer to procedure inside DLL
        Ptr_OPS_FDD_0=NULL;
        Ptr_OPS_FDD_0=(Ptr_OPS_FDD)GetProcAddress(hDLL,"OPS_FDD");
        if(Ptr_OPS_FDD_0==NULL){
            throw "FDD DLL exported function not found";
            FreeLibrary(hDLL);
        }else{
            //run the procedure inside DLL:
            Ptr_OPS_FDD_0("FDD_INPUT_Truss_Bridge_Data2_Ambient_Inch_11Channels_77824Samples_SamplingFreq_256Hz.txt",11,256);//LabScaleTruss
            //Ptr_OPS_FDD_0("FDD_INPUT_Ambient_EW_15Channels_3000Samples_SamplingFreq_20Hz.txt",15,20);//AmbientEW
            //Ptr_OPS_FDD_0("FDD_INPUT_Meriden_3Channels(3_5_8)_3686400Samples_SamplingFreq_2048Hz.txt",3,2048);//MeridenBridge
            FreeLibrary(hDLL);
        }
    }
}
于 2014-07-28T18:14:33.680 回答