服务器程序的Xamarin-Java.Interop体验(一)
這幾天需要寫一個用到Java模塊的程序,但是Java是不可能寫的,這輩子都不可能寫的,只能搞搞interop了。
目前市面上已有的基本上是IKVM.NET和JNBridgePro,后者沒太了解技術細節,前者看起來是只有單向的互操作(JVM是跑在CLR上的,或者將Java字節碼翻譯到MSIL)。
想起來之前好像說.NET 5.0要支持Java互操作,但是翻了翻dotnet/runtime庫,絲毫看不出來倉庫內在搞支持。后來就想了想,換了xamarin/java.interop庫研究看看。
按照之前Xamarin.Android的做法的話,互操作應該是雙向的。C#這邊可以繼承Java的類,然后Java那邊也會生成訪問對應C#代碼的代碼。
然后發現……他們正在支持.NET Core 3.1,但是其JNI庫引用的頭文件還是mono的,而且用到了pthread和dlfcn的頭文件,也就是說……現在必須在Linux/macOS和mono下運行。
那么先來build一下吧~
此處以Ubuntu 18.04為例。首先需要準備一些系統依賴。編譯要很久,還是選擇apt安裝吧。
sudo apt install gnupg ca-certificates sudo apt-key adv --keyserver hkp://keyserver.ubuntu.com:80 --recv-keys 3FA7E0328081BFF6A14DA29AA6A19B38D3D831EF echo "deb https://download.mono-project.com/repo/ubuntu stable-bionic main" | sudo tee /etc/apt/sources.list.d/mono-official-stable.list sudo apt update sudo apt install openjdk-8-jdk mono-devel nuget dotnet-sdk-3.1 sudo ln -s /usr/include/mono-2.0/mono /usr/include/mono?
編譯的時候TargetFrameworks要用到netcoreapp3.1,所以得安裝上。然后就是編譯內容了。
先clone一下。
git clone https://github.com/xamarin/java.interop --depth=1 cd java.interop?
然后先簡單修改一下幾個文件。
diff --git a/Directory.Build.props b/Directory.Build.props index 521e68a..1da7d44 100644 --- a/Directory.Build.props +++ b/Directory.Build.props @@ -43,6 +43,8 @@<XamarinAndroidToolsDirectory Condition=" '$(XamarinAndroidToolsDirectory)' == '' ">$(MSBuildThisFileDirectory)external\xamarin-android-tools</XamarinAndroidToolsDirectory></PropertyGroup><PropertyGroup> + <JavaCPath>/usr/lib/jvm/java-8-openjdk-amd64/bin/javac</JavaCPath> + <JarPath>/usr/lib/jvm/java-8-openjdk-amd64/bin/jar</JarPath><JavacSourceVersion Condition=" '$(JavacSourceVersion)' == '' ">1.8</JavacSourceVersion><JavacTargetVersion Condition=" '$(JavacTargetVersion)' == '' ">1.8</JavacTargetVersion><_BootClassPath Condition=" '$(JreRtJarPath)' != '' ">-bootclasspath "$(JreRtJarPath)"</_BootClassPath> diff --git a/samples/Hello/Program.cs b/samples/Hello/Program.cs index 6ffacbb..9f45fac 100644 --- a/samples/Hello/Program.cs +++ b/samples/Hello/Program.cs @@ -10,6 +10,7 @@ namespace Hellopublic static unsafe void Main (string[] args){Console.WriteLine ("Hello World!"); + JreRuntime.Initialize("/usr/lib/jvm/java-8-openjdk-amd64/jre/lib/amd64/server/libjvm.so");try {var ignore = JniRuntime.CurrentRuntime;} catch (InvalidOperationException e) { diff --git a/src/Java.Interop/Java.Interop/JniRuntime.cs b/src/Java.Interop/Java.Interop/JniRuntime.cs index 6de9021..f9fa0de 100644 --- a/src/Java.Interop/Java.Interop/JniRuntime.cs +++ b/src/Java.Interop/Java.Interop/JniRuntime.cs @@ -149,7 +149,8 @@ namespace Java.InteropDebug.Assert (count == 0);var available = GetAvailableInvocationPointers ().FirstOrDefault ();if (available == IntPtr.Zero) - throw new NotSupportedException ("No available Java runtime to attach to. Please create one."); + return null; + //throw new NotSupportedException ("No available Java runtime to attach to. Please create one.");var options = new CreationOptions () {DestroyRuntimeOnDispose = false,InvocationPointer = available, diff --git a/src/Java.Runtime.Environment/Java.Interop/JreRuntime.cs b/src/Java.Runtime.Environment/Java.Interop/JreRuntime.cs index ea1489f..9ca06b0 100644 --- a/src/Java.Runtime.Environment/Java.Interop/JreRuntime.cs +++ b/src/Java.Runtime.Environment/Java.Interop/JreRuntime.cs @@ -72,6 +72,14 @@ namespace Java.Interop {public class JreRuntime : JniRuntime{ + public static void Initialize(string path) + { + int r = NativeMethods.java_interop_jvm_load (path); + if (r != 0) { + throw new Exception ($"Could not load JVM path `{path}` ({r})!"); + } + } +static int CreateJavaVM (out IntPtr javavm, out IntPtr jnienv, ref JavaVMInitArgs args){return NativeMethods.java_interop_jvm_create (out javavm, out jnienv, ref args);?
另外,OpenJDK11應該也是可用的,不過得注意JavacSourceVersion和JavacTargetVersion=11,由于使用的部分代碼還是java8標準所以建議繼續JavacSourceVersion=1.8。暫時還沒實驗jdk11。
文件差不多編輯完了,來編譯。
make src/Java.Runtime.Environment/Java.Runtime.Environment.dll.config make mono bin/TestDebug/Hello.exe?
此時會顯示運行成功的樣子。如果沒成功,那就是我忘了哪個步驟沒寫(逃)
Hello World! Part 2! # JniEnvironment.EnvironmentPointer=94212541059552 vm.SafeHandle=140206052962432 java.lang.Object=0x55af91090e50/L hashcode=1735600054 WITHIN: GetCreatedJavaVMs: 140206052962432 POST: GetCreatedJavaVMs: 140206052962432接下來的文章將大致介紹如何在C#中直接調用Java代碼,而不是JniType一頓操作。
總結
以上是生活随笔為你收集整理的服务器程序的Xamarin-Java.Interop体验(一)的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 内存迟迟下不去,可能你就差一个GC.Co
- 下一篇: 揭秘 .NET 5 和Java 互操作