草庐IT

c# - Java .NET 互操作

coder 2024-03-23 原文

好的,我已经对此进行了一段时间的挖掘,正在寻找输入。

我需要一个可以加载和卸载本地库的 Java 应用程序,足够简单,C/C++/Java/Scripts/Executables/等等。使用 JNI 和其他内置功能不是真正的问题。

但是,我还需要加载 .NET 库的能力,这让我非常沮丧。我的第一个尝试是使用 JNI 并调用 C++ 包装器,如下所示:

java :

this.Lib = (LibHandler)Native.loadLibrary("MyLib", LibHandler.class);

菲律宾共产党:

#include <jni.h>
#using <MyLibCSharp.dll>
#include <vcclr.h>
#include <msclr\marshal.h>

using namespace msclr::interop;
using namespace System::Runtime::InteropServices;
using namespace System;

extern "C" 
{
    JNIEXPORT jstring JNICALL Java_Test(JNIEnv * env)
    { 
        marshal_context ^ context = gcnew marshal_context();
        const char* str4 = context->marshal_as<const char*>(CSharp::Class1::Test());

        jstring js = env->NewStringUTF(str4);


        return js;
    }

    JNIEXPORT jstring JNICALL Java_Test2(JNIEnv * env, jobject jobj)
    { 
        marshal_context ^ context = gcnew marshal_context();
        const char* str4 = context->marshal_as<const char*>(CSharp::Class1::Test());

        jstring js = env->NewStringUTF(str4);


        return js;
    }
}

这将不断地无法被系统加载,我可以交换文件以便 MyLib.dll 实际上是 C# 文件,并且成功加载它(但找不到任何函数,因为它不是 native C库,我不认为 .NET 可以像 C++ 那样导出),所以我没有文件位置问题。

Exception in thread "main" java.lang.UnsatisfiedLinkError: Unable to load library 'MyLib.dll': The specified module could not be found.
    at com.sun.jna.NativeLibrary.loadLibrary(NativeLibrary.java:163)
    at com.sun.jna.NativeLibrary.getInstance(NativeLibrary.java:236)
    at com.sun.jna.Library$Handler.<init>(Library.java:140)
    at com.sun.jna.Native.loadLibrary(Native.java:379)
    at com.sun.jna.Native.loadLibrary(Native.java:364)
    at EntryPoint.main(EntryPoint.java:31)

我想我会尝试将 C# 库编译为 COM 对象并以这种方式调用它,唉:

ActiveXComponent comp = new ActiveXComponent("MyLib.Class1");

失败:

Exception in thread "main" com.jacob.com.ComFailException: Can't co-create object
    at com.jacob.com.Dispatch.createInstanceNative(Native Method)
    at com.jacob.com.Dispatch.<init>(Dispatch.java:99)
    at com.jacob.activeX.ActiveXComponent.<init>(ActiveXComponent.java:58)
    at EntryPoint.main(EntryPoint.java:33)

C#代码:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Runtime.InteropServices;



namespace MyLib
{
    [Guid("4F3A0A13-4D2B-4DE6-93EA-D6861C230290"),
    ComVisible(true)]
    public interface ITest
    {
        [DispId(1)]
        string Test();
    }

    [Guid("A78C5820-3E4B-49B3-8C8E-63DD12346410"),
    InterfaceType(ComInterfaceType.InterfaceIsIDispatch),
    ComVisible(true)]
    public interface ITestEvents
    {
    }

    [Guid("3ECD46AE-E8F4-4B62-B9DC-DD7F6CB435E2"),
    ClassInterface(ClassInterfaceType.None),
    ComSourceInterfaces(typeof(ITestEvents)),
    ComVisible(true)]
    public class Class1 : ITest
    {
        public string Test()
        {
            return "This is my C# DLL COM returning a string! heh!";
        }
    }
}

我可以看到 COM 已注册,我可以用 oleview.exe 浏览它,但我不能从 vbScript 调用它...我对那个感到困惑,所以我完全没有想法,这个真是绞尽脑汁一整天。

喜欢远离 COM,但我需要保持实现相当简单(库不会由我们开发,所以我不想放弃一堆 C/C++ 代码进入 VB6 开发人员的视野)。

我很乐意回到 CLI C++ 实现方法,因为它非常简单,几乎任何人都可以做到。

任何想法将不胜感激,谢谢。

编辑:

我不能使用 System.LoadLibrary,不会让我像 Native.LoadLibrary 那样使用分配的类卸载库。

最佳答案

如果你有单一的方法来公开你的方法是合理的。明确地通过 JNI 到 C++/CLI,然后 .NET 是更可靠的解决方案,并且比尝试使用 COM 对象处理此类通信更容易维护。

无论如何,对于那些有更复杂要求的人,例如调用任何方法、传递 ref/out 参数、调用通用方法、获取设置字段、订阅 .NET 事件处理异常、获取索引属性等等。我认为自定义原生包装行不通,至少在合理的时间和成本上行不通。

如果您有这样的要求或只是想了解可能性,请查看第三方 Java 到 .NET 的桥接,例如:

如 IKVM 所述,这两个更适合这种情况。这些桥让您可以直接在 JAVA 代码中使用任何 .NET 库。它们处理所有提到的操作/场景等等,包括内置数据类型转换和已知陷阱的其他机制。

JNBridge 有点重,也更贵。但是它有一些专门用于企业系统的插件,例如 BizTalk。另一方面,Javonet 是一种非常轻便的 jar 文件解决方案,使用非常简单的 API 只需半价。另一个主要区别是 Javonet 在没有代理类的情况下工作,因此您不必生成任何包装器,您只需使用反射样式调用 .NET,而通过 JNBridge,您可以生成为您提供强类型接口(interface)的代理类,但这需要更多时间和精力.在我看来,它会稍微降低灵 active ,因为我喜欢控制引擎盖下发生的事情,并且如果您愿意,使用 Javonet 可以轻松地制作自己的强类型包装。

我认为这仍然不流行,但对于单机解决方案来说,这是一种很好的方法,可以无缝地弥补 .NET 和 JAVA 之间的差距,并具有 native 性能。它只是 web 服务所用执行时间百分比的一小部分,不需要任何高级客户端-服务器基础结构。在我的测试中,它比直接在 .NET 中执行特定代码多花费大约 30% 的时间(这非常有吸引力)。

很高兴听到你解决了你的问题,我希望这篇文章能帮助其他人解决 Java 到 .NET 的互操作问题。

您可以在下面找到使用 .NET Random 类的 Javonet 示例代码(重要的是要注意,使用第三方桥,您不仅可以访问自定义 DLL,还可以访问完整的 .NET 框架):

public void GenerateRandomNumber() throws JavonetException 
{
        NObject objRandom = Javonet.New("System.Random");
        int value = objRandom.invoke("Next",10,20);

        System.out.println(value); 
}

关于c# - Java .NET 互操作,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6378022/

有关c# - Java .NET 互操作的更多相关文章

  1. ruby-on-rails - Ruby net/ldap 模块中的内存泄漏 - 2

    作为我的Rails应用程序的一部分,我编写了一个小导入程序,它从我们的LDAP系统中吸取数据并将其塞入一个用户表中。不幸的是,与LDAP相关的代码在遍历我们的32K用户时泄漏了大量内存,我一直无法弄清楚如何解决这个问题。这个问题似乎在某种程度上与LDAP库有关,因为当我删除对LDAP内容的调用时,内存使用情况会很好地稳定下来。此外,不断增加的对象是Net::BER::BerIdentifiedString和Net::BER::BerIdentifiedArray,它们都是LDAP库的一部分。当我运行导入时,内存使用量最终达到超过1GB的峰值。如果问题存在,我需要找到一些方法来更正我的代

  2. ruby - 如何模拟 Net::HTTP::Post? - 2

    是的,我知道最好使用webmock,但我想知道如何在RSpec中模拟此方法:defmethod_to_testurl=URI.parseurireq=Net::HTTP::Post.newurl.pathres=Net::HTTP.start(url.host,url.port)do|http|http.requestreq,foo:1endresend这是RSpec:let(:uri){'http://example.com'}specify'HTTPcall'dohttp=mock:httpNet::HTTP.stub!(:start).and_yieldhttphttp.shou

  3. java - 等价于 Java 中的 Ruby Hash - 2

    我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/

  4. c# - 如何在 ruby​​ 中调用 C# dll? - 2

    如何在ruby​​中调用C#dll? 最佳答案 我能想到几种可能性:为您的DLL编写(或找人编写)一个COM包装器,如果它还没有,则使用Ruby的WIN32OLE库来调用它;看看RubyCLR,其中一位作者是JohnLam,他继续在Microsoft从事IronRuby方面的工作。(估计不会再维护了,可能不支持.Net2.0以上的版本);正如其他地方已经提到的,看看使用IronRuby,如果这是您的技术选择。有一个主题是here.请注意,最后一篇文章实际上来自JohnLam(看起来像是2009年3月),他似乎很自在地断言RubyCL

  5. C# 到 Ruby sha1 base64 编码 - 2

    我正在尝试在Ruby中复制Convert.ToBase64String()行为。这是我的C#代码:varsha1=newSHA1CryptoServiceProvider();varpasswordBytes=Encoding.UTF8.GetBytes("password");varpasswordHash=sha1.ComputeHash(passwordBytes);returnConvert.ToBase64String(passwordHash);//returns"W6ph5Mm5Pz8GgiULbPgzG37mj9g="当我在Ruby中尝试同样的事情时,我得到了相同sha

  6. java - 从 JRuby 调用 Java 类的问题 - 2

    我正在尝试使用boilerpipe来自JRuby。我看过guide从JRuby调用Java,并成功地将它与另一个Java包一起使用,但无法弄清楚为什么同样的东西不能用于boilerpipe。我正在尝试基本上从JRuby中执行与此Java等效的操作:URLurl=newURL("http://www.example.com/some-location/index.html");Stringtext=ArticleExtractor.INSTANCE.getText(url);在JRuby中试过这个:require'java'url=java.net.URL.new("http://www

  7. ruby - Net::HTTP 获取源代码和状态 - 2

    我目前正在使用以下方法获取页面的源代码:Net::HTTP.get(URI.parse(page.url))我还想获取HTTP状态,而无需发出第二个请求。有没有办法用另一种方法做到这一点?我一直在查看文档,但似乎找不到我要找的东西。 最佳答案 在我看来,除非您需要一些真正的低级访问或控制,否则最好使用Ruby的内置Open::URI模块:require'open-uri'io=open('http://www.example.org/')#=>#body=io.read[0,50]#=>"["200","OK"]io.base_ur

  8. java - 我的模型类或其他类中应该有逻辑吗 - 2

    我只想对我一直在思考的这个问题有其他意见,例如我有classuser_controller和classuserclassUserattr_accessor:name,:usernameendclassUserController//dosomethingaboutanythingaboutusersend问题是我的User类中是否应该有逻辑user=User.newuser.do_something(user1)oritshouldbeuser_controller=UserController.newuser_controller.do_something(user1,user2)我

  9. java - 什么相当于 ruby​​ 的 rack 或 python 的 Java wsgi? - 2

    什么是ruby​​的rack或python的Java的wsgi?还有一个路由库。 最佳答案 来自Python标准PEP333:Bycontrast,althoughJavahasjustasmanywebapplicationframeworksavailable,Java's"servlet"APImakesitpossibleforapplicationswrittenwithanyJavawebapplicationframeworktoruninanywebserverthatsupportstheservletAPI.ht

  10. Observability:从零开始创建 Java 微服务并监控它 (二) - 2

    这篇文章是继上一篇文章“Observability:从零开始创建Java微服务并监控它(一)”的续篇。在上一篇文章中,我们讲述了如何创建一个Javaweb应用,并使用Filebeat来收集应用所生成的日志。在今天的文章中,我来详述如何收集应用的指标,使用APM来监控应用并监督web服务的在线情况。源码可以在地址 https://github.com/liu-xiao-guo/java_observability 进行下载。摄入指标指标被视为可以随时更改的时间点值。当前请求的数量可以改变任何毫秒。你可能有1000个请求的峰值,然后一切都回到一个请求。这也意味着这些指标可能不准确,你还想提取最小/

随机推荐