我认为 Linux 的一个主要设计缺陷是在以二进制而非源代码形式分发程序时的共享对象 hell 。
这是我的具体问题:我想以 ELF 二进制形式发布一个 Linux 程序,该程序应在尽可能多的发行版上运行,以便我的强制依赖性尽可能低:在任何情况下唯一需要的库是 libpthread, libX11、librt 和 libm(当然还有 glibc)。当我使用 gcc 构建我的程序时,我正在动态链接这些库。
但是,我的程序也可以选择支持 ALSA(声音接口(interface))、Xcursor、Xfixes 和 Xxf86vm 扩展以及 GTK。但是只有当它们在用户的系统上可用时才应该使用这些,否则我的程序应该仍然运行但功能有限。例如,如果 GTK 不存在,我的程序将回退到终端模式。因为我的程序应该仍然能够在没有 ALSA、Xcursor、Xfixes 等的情况下运行。我无法动态链接到这些库,因为如果其中一个库不存在,程序将根本无法启动。
所以我需要手动检查库是否存在,然后使用 dlopen() 一个一个地打开它们,并使用 dlsym() 导入必要的函数符号。然而,这会导致各种问题:
1) 库命名约定: 共享对象通常不简单地称为“libXcursor.so”,而是具有某种版本扩展名,如“libXcursor.so.1”,甚至是非常有趣的东西,如“libXcursor.so.0.2000”。这些扩展似乎因系统而异。那么调用dlopen()应该选择哪一个呢?在这里使用硬编码名称似乎是一个非常糟糕的主意,因为名称因系统而异。因此,我想到的唯一解决方法是扫描整个库路径并查找以“libXcursor.so”前缀开头的文件名,然后进行一些自定义版本匹配。但我怎么知道它们真的兼容?
2) 库搜索路径:我到底应该在哪里寻找 *.so 文件?这也因系统而异。有一些默认路径,如/usr/lib 和/lib,但 *.so 文件也可以位于许多其他路径中。所以我必须打开/etc/ld.so.conf 并解析它以找出所有库搜索路径。这不是一件小事,因为/etc/ld.so.conf 文件也可以使用某种 include 指令,这意味着我必须解析更多的 .conf 文件,对可能的文件进行一些检查由循环 include 指令等引起的无限循环。真的没有更简单的方法来找出 *.so 的搜索路径吗?
所以,我的实际问题是:是否有更方便、更简单的方法来实现我想做的事情?创建一个具有一些可选依赖项(如 ALSA、GTK、libXcursor……但没有它也应该可以工作)的 Linux 程序真的那么复杂吗?做我想做的事有某种标准吗?还是我注定要以这种骇人听闻的方式来做?
感谢您的意见/解决方案!
最佳答案
I think a major design flaw in Linux is the shared object hell when it comes to distributing programs in binary instead of source code form.
就系统的创建者而言,这不是设计缺陷;这是一个优势——它鼓励您以源代码形式分发程序。哦,你想卖你的软件?抱歉,这不是 Linux 优化的用例。
Library naming conventions: Shared objects often aren't simply called "libXcursor.so" but have some kind of version extension like "libXcursor.so.1" or even really funny things like "libXcursor.so.0.2000".
是的,这叫做外部库版本控制。了解一下 here .从该描述中应该清楚,如果您在通常会给您 libXcursor.so.1 作为运行时引用的系统上使用 header 编译二进制文件,那么仅您兼容的共享库是 libXcursor.so.1,尝试 dlopen libXcursor.so.0.2000 将导致不可预测的崩溃。
任何提供 libXcursor.so 但不提供 libXcursor.so.1 的系统要么安装损坏,要么与您的二进制文件不兼容。
Library search paths: Where should I look for the *.so files after all?
您不应该尝试使用它们的完整路径 dlopen 这些库中的任何一个。只需调用 dlopen("libXcursor.so.1", RTLD_GLOBAL);,运行时加载器就会在系统适当的位置搜索库。
关于linux - 在运行时加载 Linux 库,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15951672/
总的来说,我对ruby还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用
我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0
在选择我想要运行操作的频率时,唯一的选项是“每天”、“每小时”和“每10分钟”。谢谢!我想为我的Rails3.1应用程序运行调度程序。 最佳答案 这不是一个优雅的解决方案,但您可以安排它每天运行,并在实际开始工作之前检查日期是否为当月的第一天。 关于ruby-如何每月在Heroku运行一次Scheduler插件?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/8692687/
exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby中使用两个参数异步运行exe吗?我已经尝试过ruby命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何rubygems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除
我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r
鉴于我有以下迁移: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
Sinatra新手;我正在运行一些rspec测试,但在日志中收到了一堆不需要的噪音。如何消除日志中过多的噪音?我仔细检查了环境是否设置为:test,这意味着记录器级别应设置为WARN而不是DEBUG。spec_helper:require"./app"require"sinatra"require"rspec"require"rack/test"require"database_cleaner"require"factory_girl"set:environment,:testFactoryGirl.definition_file_paths=%w{./factories./test/
我收到这个错误:RuntimeError(自动加载常量Apps时检测到循环依赖当我使用多线程时。下面是我的代码。为什么会这样?我尝试多线程的原因是因为我正在编写一个HTML抓取应用程序。对Nokogiri::HTML(open())的调用是一个同步阻塞调用,需要1秒才能返回,我有100,000多个页面要访问,所以我试图运行多个线程来解决这个问题。有更好的方法吗?classToolsController0)app.website=array.join(',')putsapp.websiteelseapp.website="NONE"endapp.saveapps=Apps.order("
GivenIamadumbprogrammerandIamusingrspecandIamusingsporkandIwanttodebug...mmm...let'ssaaay,aspecforPhone.那么,我应该把“require'ruby-debug'”行放在哪里,以便在phone_spec.rb的特定点停止处理?(我所要求的只是一个大而粗的箭头,即使是一个有挑战性的程序员也能看到:-3)我已经尝试了很多位置,除非我没有正确测试它们,否则会发生一些奇怪的事情:在spec_helper.rb中的以下位置:require'rubygems'require'spork'
我一直致力于让我们的Rails2.3.8应用程序在JRuby下正确运行。一切正常,直到我启用config.threadsafe!以实现JRuby提供的并发性。这导致lib/中的模块和类不再自动加载。使用config.threadsafe!启用:$rubyscript/runner-eproduction'pSim::Sim200Provisioner'/Users/amchale/.rvm/gems/jruby-1.5.1@web-services/gems/activesupport-2.3.8/lib/active_support/dependencies.rb:105:in`co