草庐IT

java - 我的 dll 代码可以从 exe 文件运行,但无法从 Java loadLibrary 加载

coder 2023-08-28 原文

我创建了一个 C++ 模块来构建到一个共享库文件中,然后使用 JNI 从 Java 调用它。

我有 2 个环境,Windows 和 Unix,我有一个 C++ 可执行程序和一个 Java 程序,我只是为每个环境重新编译。

  • 当我在 Unix 中编译我的 tester.exe 程序并使用方法运行它时
    从我的图书馆(.so)它工作正常。
  • 当我在 Unix 中编译我的 Java 程序并加载我的库 (.so) 时
    Java 的 loadLibrary,它工作正常。
  • 当我在 Windows 中编译我的 tester.exe 程序并使用它运行时
    我的库 (.dll) 中的方法工作正常。就像unix一样
    版本。
  • 当我在 Windows 中编译我的 Java 程序并使用 Java 的 loadLibrary 加载我的库 (.dll) 时,它失败了。它说尝试访问
    无效地址。

  • 我无法弄清楚为什么它在 Windows 中运行时不能与 Java loadLibrary 一起使用,但它可以在其他任何地方使用相同的代码运行。如果我延迟加载我的库使用的依赖 DLL,那么我的库会以 Java 加载但无法正常工作。我知道有特定的代码会导致 Java 加载我的库时出现问题,但我不明白为什么我的 C++ exe 对相同的方法和库没有问题。

    我的 dll 有 1 个公开的方法,它从一些现有的库中调用 4 个方法。如果我将这 4 个方法注释掉,那么我的 dll 就可以在 Java 中正常加载。我知道这与我的 dll 链接到的库中的这些方法有关。 Java 看待依赖库的方式有什么不同吗?我已经尝试先加载依赖库,但是我加载的其中一个 dll 文件导致递归错误和堆栈溢出。

    任何人都知道解决 DLL 导致递归错误导致堆栈溢出的方法吗?我需要其中的方法,但我无法使用 java loadLibrary 加载它。

    以下是有关所涉及文件和实际错误消息的更多详细信息。
    我在我的初始 dll 文件中添加了一个 DllMain 只是为了查看加载的内容和时间。如果我将相同的程序 (my_plain_dll_to_call_JNI_DLL) 编译为 exe 文件,则一切正常。如果我编译它并从我的 java 程序加载它,就会发生这种情况。
  • myJavaProgram,只需调用 System.loadLibrary() 即可加载基本的 .dll
    调用包含 JNI 代码的其他 dll 中的方法的文件。
  • my_plain_dll_to_call_JNI_DLL 是我通过将它链接到我的
    dll库文件只是为了测试依赖关系。它只是调用一个方法
    来自另一个正在调用我需要的 native 代码的dll。
  • my_JNI_DLL.ll 是一个与现有 C++ 编程链接的 dll 文件
    我需要从 JNI 访问的库。它包含直接调用
    现有源代码库中的方法。

  • 我写了显示每行左侧文本的文件名,以显示执行所在的层。
    c:\java myJavaProgram myJavaProgram: Java Static Method Entry. myJavaProgram: Java Calling System.loadLibrary(my_plain_dll_to_call_JNI_DLL) my_JNI_DLL.dll: Entering DllMain my_JNI_DLL.dll: DLL_PROCESS_ATTACH my_plain_dll_to_call_JNI_DLL: DLL_PROCESS_ATTACH my_plain_dll_to_call_JNI_DLL: DLL_THREAD_ATTACH my_plain_dll_to_call_JNI_DLL: DLL_THREAD_DETACH my_plain_dll_to_call_JNI_DLL: DLL_PROCESS_DETACH myJavaProgram: my_plain_dll_to_call_JNI_DLL Loaded! myJavaProgram: Java Static Method Exit. myJavaProgram: Entering Main(). my_plain_dll_to_call_JNI_DLL: In call_my_JNI_DLL_method my_JNI_DLL.dll: In my_JNI_DLL_method my_JNI_DLL.dll: Entering my_JNI_DLL_CheckEnvironmentVariables() my_JNI_DLL.dll: Exiting my_JNI_DLL_CheckEnvironmentVariables my_JNI_DLL.dll: Calling StartExistingNativeCode. # # A fatal error has been detected by the Java Runtime Environment: # # Internal Error (0xc0fb007e), pid=7500, tid=7552 # # JRE version: 6.0_21-b06 # Java VM: Java HotSpot(TM) Client VM (17.0-b16 mixed mode, sharing windows-x86 ) # Problematic frame: # C [KERNELBASE.dll+0x9673] # # An error report file with more information is saved as: # C:\hs_err_pid7500.log # # If you would like to submit a bug report, please visit: # http://java.sun.com/webapps/bugreport/crash.jsp # The crash happened outside the Java Virtual Machine in native code. # See problematic frame for where to report the bug. # my_plain_dll_to_call_JNI_DLL: DLL_PROCESS_DETACH my_JNI_DLL.dll: Entering DllMain my_JNI_DLL.dll DLL_PROCESS_DETACH

    更新
    我已将问题缩小到从我的程序使用的另一个 dll 链接的内存管理库。它使用的 dll 是 sh33w32.dll,它被称为 SmartHeap,我认为是一家名为 Microquil 的公司。我有 3.3 版,当 Java LoadLibrary 尝试加载该 dll 时,它失败了。我不确定我可以做些什么来让 java 处理加载该库。它必须与 Java 可以访问的内存区域有关,而不是 Windows 允许 exe 访问的内容。 exe用SmartHeap库没问题,但是Java不允许我用。任何想法或经验处理这个?我试图通过重新编译其他库来删除链接库,但是代码中的正常调用失败了,但通常可以正常工作。

    找到附加信息
    在java中加载失败的dll中的函数称为MemRegisterTask。它来自 Microquill 的名为 SmartHeap 的产品。这是我找到的有关此功能的文档。我认为这种内存分配是导致 java 无法加载它的原因。

    MemRegisterTask 初始化 SmartHeap 库。在大多数平台上,您不需要调用 MemRegisterTask,因为 SmartHeap 会在您第一次调用时自行初始化。

    SmartHeap 为每个任务或进程维护一个注册引用计数。每次调用 MemRegisterTask 时,此引用计数都会增加。如果最后一次调用 SmartHeap 发生在应用程序准备终止之前,则可以调用 MemUnregisterTask 来终止 SmartHeap。 MemUnregisterTask 将注册引用计数减一——当计数为零时,SmartHeap 将释放与当前任务或进程关联的任何 SmartHeap 分配的内存和调试状态。

    最佳答案

    hs_err ... 日志文件中的任何有用信息。通常有一个堆栈回溯等。
    指出某事。

    还尝试在里面运行 java.exe(带有运行加载内容的测试的参数)
    调试器?

    从上面的跟踪可以看到加载似乎工作正常(跟踪输出表明
    dllentrypoint/dllmain 已通过您的跟踪输出进行了扩充)。

    加载顺序如下:

  • 加载依赖的 dll
  • 加载 dll 本身
  • 调用带有进程附加的 dllentrypoint/dllmain

  • 所以这已经超出了加载 DLL 的范围。

    您是否检查过是否正在使用 Windows 的调试/发布运行时?调试可能与发布冲突 - Java 是发布,您的示例 exe 可能与您的 dll 构建相同。

    关于java - 我的 dll 代码可以从 exe 文件运行,但无法从 Java loadLibrary 加载,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10021786/

    有关java - 我的 dll 代码可以从 exe 文件运行,但无法从 Java loadLibrary 加载的更多相关文章

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

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

    2. ruby - 使用 RubyZip 生成 ZIP 文件时设置压缩级别 - 2

      我有一个Ruby程序,它使用rubyzip压缩XML文件的目录树。gem。我的问题是文件开始变得很重,我想提高压缩级别,因为压缩时间不是问题。我在rubyzipdocumentation中找不到一种为创建的ZIP文件指定压缩级别的方法。有人知道如何更改此设置吗?是否有另一个允许指定压缩级别的Ruby库? 最佳答案 这是我通过查看ruby​​zip内部创建的代码。level=Zlib::BEST_COMPRESSIONZip::ZipOutputStream.open(zip_file)do|zip|Dir.glob("**/*")d

    3. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

      类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

    4. ruby - 其他文件中的 Rake 任务 - 2

      我试图在一个项目中使用rake,如果我把所有东西都放到Rakefile中,它会很大并且很难读取/找到东西,所以我试着将每个命名空间放在lib/rake中它自己的文件中,我添加了这个到我的rake文件的顶部:Dir['#{File.dirname(__FILE__)}/lib/rake/*.rake'].map{|f|requiref}它加载文件没问题,但没有任务。我现在只有一个.rake文件作为测试,名为“servers.rake”,它看起来像这样:namespace:serverdotask:testdoputs"test"endend所以当我运行rakeserver:testid时

    5. ruby-on-rails - 在 Rails 中将文件大小字符串转换为等效千字节 - 2

      我的目标是转换表单输入,例如“100兆字节”或“1GB”,并将其转换为我可以存储在数据库中的文件大小(以千字节为单位)。目前,我有这个:defquota_convert@regex=/([0-9]+)(.*)s/@sizes=%w{kilobytemegabytegigabyte}m=self.quota.match(@regex)if@sizes.include?m[2]eval("self.quota=#{m[1]}.#{m[2]}")endend这有效,但前提是输入是倍数(“gigabytes”,而不是“gigabyte”)并且由于使用了eval看起来疯狂不安全。所以,功能正常,

    6. ruby-on-rails - Rails 3 中的多个路由文件 - 2

      Rails2.3可以选择随时使用RouteSet#add_configuration_file添加更多路由。是否可以在Rails3项目中做同样的事情? 最佳答案 在config/application.rb中:config.paths.config.routes在Rails3.2(也可能是Rails3.1)中,使用:config.paths["config/routes"] 关于ruby-on-rails-Rails3中的多个路由文件,我们在StackOverflow上找到一个类似的问题

    7. ruby - 将差异补丁应用于字符串/文件 - 2

      对于具有离线功能的智能手机应用程序,我正在为Xml文件创建单向文本同步。我希望我的服务器将增量/差异(例如GNU差异补丁)发送到目标设备。这是计划:Time=0Server:hasversion_1ofXmlfile(~800kiB)Client:hasversion_1ofXmlfile(~800kiB)Time=1Server:hasversion_1andversion_2ofXmlfile(each~800kiB)computesdeltaoftheseversions(=patch)(~10kiB)sendspatchtoClient(~10kiBtransferred)Cl

    8. ruby - 如何将脚本文件的末尾读取为数据文件(Perl 或任何其他语言) - 2

      我正在寻找执行以下操作的正确语法(在Perl、Shell或Ruby中):#variabletoaccessthedatalinesappendedasafileEND_OF_SCRIPT_MARKERrawdatastartshereanditcontinues. 最佳答案 Perl用__DATA__做这个:#!/usr/bin/perlusestrict;usewarnings;while(){print;}__DATA__Texttoprintgoeshere 关于ruby-如何将脚

    9. ruby - 如何在 buildr 项目中使用 Ruby 代码? - 2

      如何在buildr项目中使用Ruby?我在很多不同的项目中使用过Ruby、JRuby、Java和Clojure。我目前正在使用我的标准Ruby开发一个模拟应用程序,我想尝试使用Clojure后端(我确实喜欢功能代码)以及JRubygui和测试套件。我还可以看到在未来的不同项目中使用Scala作为后端。我想我要为我的项目尝试一下buildr(http://buildr.apache.org/),但我注意到buildr似乎没有设置为在项目中使用JRuby代码本身!这看起来有点傻,因为该工具旨在统一通用的JVM语言并且是在ruby中构建的。除了将输出的jar包含在一个独特的、仅限ruby​​

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

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

    随机推荐