草庐IT

c++ - 从另一个共享对象动态加载共享对象时的约束?

coder 2023-06-19 原文

我正在从 main 动态加载(使用 dlopen())一个共享对象(名为 libprofile1.so)。

libprofile1.so 中,我定义了工厂函数 CreateProfile 和类 ProfileCreateProfile 函数创建一个 Profile 类的实例并返回指向它的指针。类 Profile 有一个方法 pMethod

在 main 中,加载 libprofile1.so 后,我调用 CreateProfile 方法,该方法返回指向 Profile 类对象的指针(称它为 p)。
之后,我针对对象 p (p->pMethod) 调用 pMethod 方法。在此方法中,我动态加载其他共享对象 (libdatasources.so)。

在这个共享对象中,我有一个工厂函数 CreateDataSource 和类 DataSource
CreateDataSource 函数创建一个 DataSource 类的实例并返回指向它的指针。 DataSource 类有方法 dsMethod

如您所见,两个共享对象的结构相似。

pMethod 加载 libdatasources.so 后,我正在调用 CreateDataSource 方法,它返回一个指向 实例的指针DataSource 类,称之为ds。 然后我调用 ds 对象的 dsMethod
(ds->dsMethod).


现在,问题来了。

当我尝试调用 ds 对象的 dsMethod 时,我首先加载的共享对象 (libprofile1.so) 没有加载。实际上 dlopen() 返回 NULL。当我在 dlopen 之后阅读 dlerror 时,我得到:

./libprofile1.so: undefined symbol :_ZN18DataSource13dsMethod

因此,如果我调用 ds->Method,那么第一个共享对象不会加载!
如果我从源代码中注释掉调用 ds->dsMethod,那么我的 libprofile1.solibdatasources.so 加载时没有任何问题。< br/=""> 我没有看到从第二个 SO 调用方法与首先加载 SO 之间的联系???

也许我不知道,但是从一个也被动态加载的共享对象动态加载一个共享对象时是否有任何限制?

顺便说一句,dlopenRTLD_NOW|RTLD_GLOBAL 一起使用。我尝试使用 RTLD_LAZY,但仍然是同样的问题。

更新:

库是在 Eclipse 中构建的。两个库的 G++ 编译器和链接器选项相同。
这里是 G++ 编译器:

-O0 -g3 -Wall -c -fmessage-length=0

和 G++ 链接器:

-shared

选项,粘贴自 Project Properties -> Settings -> Tool Settings

提前致谢。

最佳答案

正如 Martin York 所指出的,这就是它在 Linux 中的工作方式。当链接到一个库时,你也必须链接到所有依赖项。这在 Windows 中是不同的,DLL 会自己处理它们的依赖关系。动态加载具有另一个库作为依赖项的库时,您需要首先使用 RTLD_GLOBAL 标志加载该库。恕我直言,这很尴尬,因为您可能无法知道另一个共享对象需要哪些依赖项,或者依赖项可能会随着二进制兼容的新版本而改变。据我所知(以及阅读 g++ 和 ld 联机帮助页),不可能创建类似于 Windows 的 DLL 的行为。 这是一个小测试用例:

两个.cpp:

#include <iostream>

extern "C"
{
   void bar()
   {
      std::cout << "bar()\n";
   }
}

一个.cpp:

#include <iostream>

extern "C"
{
   void bar();

   void foo()
   {
      std::cout << "foo()\n";
      bar ();
   }
}

测试.cpp:

#include <dlfcn.h>
#include <iostream>

int main (int argc, char *argv[])
{
   using namespace std;
//       void *libtwo = dlopen("./libtwo.so", RTLD_NOW | RTLD_GLOBAL);
   void *libone = dlopen("./libone.so", RTLD_NOW);
   if (!libone)
   {
      cout << "dlopen(libone.so) failed: " << dlerror() << "\n";
      return 1;
   }
   typedef void (*foo_t)();
   foo_t foo = reinterpret_cast<foo_t>(dlsym(libone, "foo"));
   if (!foo)
   {
      cout << "dlsym(libone.so, foo) failed\n";
      return 2;
   }
}

one.cpp在libtwo.so中编译成libone.so和two.cpp,test.cpp在test中编译成二进制。 这将失败,只有在注释行未被注释时才会成功。

关于c++ - 从另一个共享对象动态加载共享对象时的约束?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3601487/

有关c++ - 从另一个共享对象动态加载共享对象时的约束?的更多相关文章

  1. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  2. ruby-on-rails - 按天对 Mongoid 对象进行分组 - 2

    在控制台中反复尝试之后,我想到了这种方法,可以按发生日期对类似activerecord的(Mongoid)对象进行分组。我不确定这是完成此任务的最佳方法,但它确实有效。有没有人有更好的建议,或者这是一个很好的方法?#eventsisanarrayofactiverecord-likeobjectsthatincludeatimeattributeevents.map{|event|#converteventsarrayintoanarrayofhasheswiththedayofthemonthandtheevent{:number=>event.time.day,:event=>ev

  3. ruby - 使用 Vim Rails,您可以创建一个新的迁移文件并一次性打开它吗? - 2

    使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta

  4. ruby-on-rails - Rails - 一个 View 中的多个模型 - 2

    我需要从一个View访问多个模型。以前,我的links_controller仅用于提供以不同方式排序的链接资源。现在我想包括一个部分(我假设)显示按分数排序的顶级用户(@users=User.all.sort_by(&:score))我知道我可以将此代码插入每个链接操作并从View访问它,但这似乎不是“ruby方式”,我将需要在不久的将来访问更多模型。这可能会变得很脏,是否有针对这种情况的任何技术?注意事项:我认为我的应用程序正朝着单一格式和动态页面内容的方向发展,本质上是一个典型的网络应用程序。我知道before_filter但考虑到我希望应用程序进入的方向,这似乎很麻烦。最终从任何

  5. ruby-on-rails - 渲染另一个 Controller 的 View - 2

    我想要做的是有2个不同的Controller,client和test_client。客户端Controller已经构建,我想创建一个test_clientController,我可以使用它来玩弄客户端的UI并根据需要进行调整。我主要是想绕过我在客户端中内置的验证及其对加载数据的管理Controller的依赖。所以我希望test_clientController加载示例数据集,然后呈现客户端Controller的索引View,以便我可以调整客户端UI。就是这样。我在test_clients索引方法中试过这个:classTestClientdefindexrender:template=>

  6. ruby-on-rails - 如何优雅地重启 thin + nginx? - 2

    我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server

  7. ruby - 如何在续集中重新加载表模式? - 2

    鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende

  8. ruby-on-rails - 如何验证非模型(甚至非对象)字段 - 2

    我有一个表单,其中有很多字段取自数组(而不是模型或对象)。我如何验证这些字段的存在?solve_problem_pathdo|f|%>... 最佳答案 创建一个简单的类来包装请求参数并使用ActiveModel::Validations。#definedsomewhere,atthesimplest:require'ostruct'classSolvetrue#youcouldevencheckthesolutionwithavalidatorvalidatedoerrors.add(:base,"WRONG!!!")unlesss

  9. Ruby 写入和读取对象到文件 - 2

    好的,所以我的目标是轻松地将一些数据保存到磁盘以备后用。您如何简单地写入然后读取一个对象?所以如果我有一个简单的类classCattr_accessor:a,:bdefinitialize(a,b)@a,@b=a,bendend所以如果我从中非常快地制作一个objobj=C.new("foo","bar")#justgaveitsomerandomvalues然后我可以把它变成一个kindaidstring=obj.to_s#whichreturns""我终于可以将此字符串打印到文件或其他内容中。我的问题是,我该如何再次将这个id变回一个对象?我知道我可以自己挑选信息并制作一个接受该信

  10. ruby - 通过 ruby​​ 进程共享变量 - 2

    我正在编写一个gem,我必须在其中fork两个启动两个webrick服务器的进程。我想通过基类的类方法启动这个服务器,因为应该只有这两个服务器在运行,而不是多个。在运行时,我想调用这两个服务器上的一些方法来更改变量。我的问题是,我无法通过基类的类方法访问fork的实例变量。此外,我不能在我的基类中使用线程,因为在幕后我正在使用另一个不是线程安全的库。所以我必须将每个服务器派生到它自己的进程。我用类变量试过了,比如@@server。但是当我试图通过基类访问这个变量时,它是nil。我读到在Ruby中不可能在分支之间共享类变量,对吗?那么,还有其他解决办法吗?我考虑过使用单例,但我不确定这是

随机推荐