簡體   English   中英

使用 JNI 從 Java 調用 C# 方法

[英]Call C# method from Java using JNI

我試圖按照我在此處找到的示例進行操作,但無法使其正常工作。 這是我的 Java 課程

package jniTester;
public class JNITester {
    static {
        System.load("D:\\\\VisualStudio_Cpp_2017\\SkriptumTeil5\\Debug\\HelloWorldJNI.dll");
    }

    public static native String welcome(String name);
}

從此我用 javah 創建了 jniTester.h 文件

這是我的 C# 類

namespace HelloWorldJNI
{

    public static class HelloWorldJNI
    {
        public static String Welcome(String name)
        {
            return "Hello " + name + "! This is your C# buddy.";
        }
    }
}

從此我創建了 HelloWorldJNI.netmodule

這是我的 cpp 課

#include "stdafx.h"
#include <jni.h>
#include <string>
#include "jniTester.h"
#using "D:\VisualStudio_C#_2017\SkriptumTeil5\HelloWorldJNI\HelloWorldJNI.netmodule"


using namespace std;

JNIEXPORT jstring JNICALL Java_jniTester_JNITester_welcome(JNIEnv *env, jclass thisclass, jstring inJNIStr) {
    // Step 1: Convert the JNI String (jstring) into C-String (char*)
    const char *inCStr = env->GetStringUTFChars(inJNIStr, NULL);
    if (NULL == inCStr) return NULL;

    // Step 2: Convert the C++ string to C-string, then to JNI String (jstring) and return
    //string outCppStr = "Hello " + std::string(inCStr) + ". Greetings from your C++ buddy";
    //env->ReleaseStringUTFChars(inJNIStr, inCStr);  // release resources
    //return env->NewStringUTF(outCppStr.c_str());

    //// Alternate Step 2:
    System::String^ outStr = HelloWorldJNI::HelloWorldJNI::Welcome(gcnew System::String(inCStr));
    env->ReleaseStringUTFChars(inJNIStr, inCStr);  // release resources
    char* converted = static_cast<char*>((System::Runtime::InteropServices::Marshal::StringToHGlobalAnsi(outStr)).ToPointer());
    return env->NewStringUTF(converted);
 }

步驟 2 下的代碼有效。 但是,這不是調用我的 C# 方法。 替代步驟 2 下的實施失敗

# A fatal error has been detected by the Java Runtime Environment:
#
#  Internal Error (0xe0434352), pid=37224, tid=0x00003350

我不是 cpp 專家,所以我完全一無所知。 這里有什么問題?

首先,您需要使用 COM 接口在 C# 中創建程序集 DLL。

然后您可以制作一個本機(非托管)DLL 包裝庫,您可以將其與 JNI 一起使用。

您可以按照本指南從非托管 C 代碼調用托管 NET COM 對象

您也可以查看JNI 手冊

一個簡短的例子:

C# 代碼 Managed.cs

using System;
using System.Reflection;
using System.Runtime.InteropServices;

[assembly: ComVisible(true)] 
[assembly: AssemblyDelaySign(false)] 
[assembly:AssemblyKeyFileAttribute("Managed.snk")]
[assembly: AssemblyVersion("1.0.0.0")]
namespace ManagedClassLibrary {

    [Guid("1CE4ECCB-EB78-4A50-8781-444052C7AEAE")]
    [ComVisible(true)]
    public interface IArithmetic {
        int sum(int lsh, int rhs);
        int subtract(int lsh, int rhs);
    }
    [Guid("30F078F9-F161-4112-B61A-B3BD6B63CB4C"), 
        ClassInterface(ClassInterfaceType.None), 
        ComSourceInterfaces(typeof(IArithmetic))
    ]
    [ComVisible(true)]
    public class ArithmeticImpl:IArithmetic {

        public int sum(int lsh, int rhs)
        {
             return lsh + rhs;
        } 
        public int subtract(int lsh, int rhs)
        {
             return lsh - rhs;
        }
    }

}

C++ dll 包裝器代碼 dotnet_arithmetic.cpp

#include <windows.h>
#include <Objbase.h>

#include <jni.h>

#import "Managed.tlb" named_guids raw_interfaces_only

using namespace Managed;

class DNArithmetic 
{
    DNArithmetic(const DNArithmetic&) = delete;
    DNArithmetic& operator=(const DNArithmetic&) = delete;
private:
    DNArithmetic() noexcept
    {
     mi_.CreateInstance(CLSID_ArithmeticImpl);
    }
public:
    const DNArithmetic* instanse() {
        if(!_co_init) { 
         ::CoInitialize(NULL);
         _co_init = true;
        }
        static DNArithmetic _ret;            
        return &_ret;
    }
    int sum(int lsh, int rhs) const {
        int ret;
        mi_->sum(lsh, rhs, &ret);
        return ret;
    }

    int subtract(int lsh, int rhs) const {
        int ret;
        mi_->subtract(lsh, rhs, &ret);
        return ret;
    }

    ~DNArithmetic() noexcept
    { 
      ::CoUninitialize();           
    }
private:
    IArithmeticPtr mi_;
    static bool _co_init;
};

bool DNArithmetic::_co_init = false;

extern "C" {

    JNIEXPORT jint JNICALL Java_Arithmetic_dotnet_1sum(JNIEnv *evn, jclass clazz, jint lsh, jint rhs)
    {
        return DNArithmetic::instance()->sum(lsh, rhs);
    }

    JNIEXPORT jint JNICALL Java_Arithmetic_dotnet_1subtract(JNIEnv *evn, jclass clazz, jint lsh, jint rhs)
    {
        return DNArithmetic::instance()->subtract(lsh, rhs);
    }

    BOOL WINAPI DllMain(::HMODULE hprocess,::DWORD fdwReason,::LPVOID lpvReserved)
    {
        switch (fdwReason) {
        case DLL_PROCESS_ATTACH:           
            break;
        case DLL_PROCESS_DETACH:                
            break;
        case DLL_THREAD_ATTACH:
           break;
        case DLL_THREAD_DETACH:
           break;
        }
        return TRUE; // successful
    }

}

Java 代碼 Arithmetic.java

public class Arithmetic {

    private static native int dotnet_sum(int lsh, int rhs);

    private static native int dotnet_subtract(int lsh, int rhs);

    private Arithmetic() {
        try {
            System.loadLibrary("dotnet_arithmetic.dll");
        } catch (UnsatisfiedLinkError e) {
            throw new IllegalStateException(e);
        }
    }

    private int sum(int lsh, int rhs) {
        return  dotnet_sum(lsh, rhs);
    }

    private int subtract(int lsh, int rhs) {
        return dotnet_subtract(lsh,rhs);
    }

    public static void main(String[] args) {
        System.setProperty("java.library.path", System.getProperty("user.dir") );
        System.out.println("About to call C# functions from Java over the MS COM");
        Arithmetic instance = new Arithmetic();
        System.out.println("C# 1 + 2 = " + instance.sum(1,2) );
        System.out.println("C# 2 - 1 = " + instance.subtract(2,1) );
    }
}

和一個構建文件 build.cmd (用於演示的簡單 bat 文件建議)

@echo off
rem make sure you have JAVA_HOME system variable set
rem Please set your VS root dir, should be  somewhere in C:\Program Files 
SET VS_HOME= 
echo %VS_HOME%

echo Take Visual studio command line tools
    call %VS_HOME%\VC\Auxiliary\Build\vcvars64.bat

echo Build C# class library
    sn /k Managed.snk
    csc /optimize /target:library /out:Managed.DLL Managed.cs
    RegAsm  Managed.DLL /tlb:Managed.tlb
    gacutil /i Managed.DLL 

echo Build C++ wrapper JNI DLL
    cl /c /nologo /GL /Zl /std:c++latest /I%JAVA_HOME%\include /I%JAVA_HOME%\include\win32 /Fodotnet_arithmetic.obj dotnet_arithmetic.cpp
    link /DLL /LTCG /LIBPATH:%JAVA_HOME%\lib /OUT:dotnet_arithmetic.dll msvcrt.lib kernel32.lib Ole32.lib jvm.lib dotnet_arithmetic.obj

echo Build Java

javac Arithmetic.java

echo Running the Demo

java Arithmetic

pause

注意:此解決方案僅適用於 Windows 操作系統。 並且不適用於 Unix 上的 Mono。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM