c++ - 使用 dlopen 时,我应该链接到我打开的库吗?

标签 c++ linux cmake shared-libraries dlopen

我想我在使用 dlopen 时不必链接到共享库。但是,在 cmake target_link_libraries(main_dlopen dl) 中导致链接器错误

main_dlopen.cpp.o: In function `main':
main_dlopen.cpp:25: undefined reference to `ntclass::init(char const*)'
etc...

如果我使用 target_link_libraries(main_dlopen dl ntclass),其中 libntclass.so 是我的库,那么一切都很好。

它真的很好还是我遗漏了什么?作为背景,我想测试非线程安全库,如解释的那样 here ,并期望应避免使用非线程安全库进行链接。 下面部分回答了自己

完整示例如下(使用 this 作为引用)。

(共享库)

nt类.h

#ifndef NTCLASS_H
#define NTCLASS_H

#include <cstddef>

class ntclass
{
  private:
    static char *sptr;
    char *ptr;

  public:
    ntclass() : ptr(NULL) {}
    ~ntclass();

    void init(const char* str);
    void print();
};

typedef ntclass* create_t();

#endif // NTCLASS_H

ntclass.cpp

#include "ntclass.h"
#include <stdio.h>
#include <string.h>
#include <iostream>

char *gptr = NULL;

char *ntclass::sptr = NULL;

ntclass::~ntclass()
{
  if (gptr)
  {
    delete[] gptr;
    gptr = NULL;
  }
  if (sptr)
  {
    delete[] sptr;
    sptr = NULL;
  }
  if (ptr)
  {
    delete[] ptr;
    ptr = NULL;
  }
}

void ntclass::init(const char* str)
{
  int size = strlen(str)*sizeof(char);
  gptr = new char[size];
  memcpy(gptr, str, size);
  sptr = new char[size];
  memcpy(sptr, str, size);
  ptr = new char[size];
  memcpy(ptr, str, size);
}

void ntclass::print()
{
  std::cout << "Global: " << gptr << std::endl;
  std::cout << "Static: " << sptr << std::endl;
  std::cout << "Normal: " << ptr << std::endl;
}

extern "C" ntclass *create()
{
  return new ntclass();
}

(主要可执行文件)

main_dlopen.cpp

#include <iostream>
#include "ntclass.h"

#include <dlfcn.h>
#include <stdlib.h>

using namespace std;

int main()
{
  void *handle = dlopen("./libntclass.so", RTLD_NOW);

  if (handle == NULL)
  {
     cerr << dlerror() << endl;
     exit(-1);
  }

  create_t *createA = (create_t*) dlsym(handle, "create");
  create_t *createB = (create_t*) dlsym(handle, "create");

  ntclass *A = createA();
  ntclass *B = createB();

  A->init("A");
  B->init("B");

  A->print();
  B->print();

  delete A;
  delete B;

  return 0;
}

(命令)

cmake_minimum_required(VERSION 2.8)

set( CMAKE_VERBOSE_MAKEFILE on )
set(CMAKE_BUILD_TYPE RelWithDebInfo)

add_library(ntclass SHARED ntclass.cpp)

add_executable(main_dlopen main_dlopen.cpp)
target_link_libraries(main_dlopen dl) # <-- Here is a problem

部分答案: 我为 ntclass 方法(init、print、~ntclass)添加了关键字 virtual,现在它工作正常。尽管如此,谁能解释为什么需要它?

最佳答案

您的“部分答案”是正确的解决方案。有关说明,请参阅 that nice answer关于 virtual 关键字。

简而言之:

直到ntclass中的init方法被声明为virtual,表达式

A->init("A")

使用方法的定义在那个类(独立于实际类型对象A有)。并且由于 main_dlopen.cpp 中不存在此定义,因此链接器会生成错误。

使用 virtual 关键字,当A 对象的实际类型 时,init 方法的解析被推迟到运行时会被知道。

关于c++ - 使用 dlopen 时,我应该链接到我打开的库吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44302741/

相关文章:

linux - 除了使用 Ctrl + C 之外,如何在 linux 中停止正在运行的 R 命令?

c - 如何确保生成的 config.h 文件在包含路径中?

c++ - 更新 CMake(似乎)破坏了我的程序

c++ - OpenCV 3.0 中缺少 cv::Mat.refcount

linux - 如何使我的自定义基于 Debian 运行?遇到一些问题

c++ - 函数/方法如何返回对对象的引用在内部工作

linux - Bash 脚本 - 运行应用程序的当前或新实例

c++ - 如何在 cmake 中使用 FFTW 库?

c++ - OpenMP 并行线程

c++ - 如何在 C++ 中创建包含从 0 到 N 的元素的行 vector ?