发布时间:2025-01-08 20:07:42
Go语言(Golang)是由谷歌公司推出的一种开源编程语言,它具有优秀的并发性能、高效的内存管理和简洁的语法等特点。尽管Go语言本身已经具备了很强大的功能和性能,但有时候我们还是需要与其他编程语言进行交互,比如与Java进行合作开发。在本文中,我们将探讨如何在Golang中运行Java代码。
Go语言的中CGO是一个非常强大的工具,它可以方便地实现C/C++语言的调用。而Java Native Interface(JNI)是Java虚拟机(JVM)提供的一套API,允许Java代码调用其他语言编写的代码。所以,我们可以通过CGO调用JNI来实现在Golang中运行Java代码。
首先,我们需要编写一个C语言的桥接文件,将其保存为bridge.c
:
#include <jni.h>
#include <stdio.h>
JNIEXPORT void JNICALL Java_MyClass_sayHello(JNIEnv *env, jobject obj) {
printf("Hello from Java!\n");
}
接下来,我们需要编写一个Java类MyClass
,并在其中定义一个native
方法:
public class MyClass {
native void sayHello();
public static void main(String[] args) {
System.loadLibrary("bridge");
new MyClass().sayHello();
}
}
接下来,我们在命令行中使用javac
命令将Java文件编译为字节码文件:
$ javac MyClass.java
然后,我们可以使用javah
命令生成Java头文件:
$ javah -jni MyClass
接着,我们可以将生成的头文件中的方法实现拷贝到我们的bridge.c
中:
#include <jni.h>
#include <stdio.h>
JNIEXPORT void JNICALL Java_MyClass_sayHello(JNIEnv *env, jobject obj) {
printf("Hello from Java!\n");
}
void JNICALL Java_MyClass_sayHello(JNIEnv *env, jobject obj);
int main() {
Java_MyClass_sayHello(NULL, NULL);
return 0;
}
然后,我们可以使用gcc
命令将C代码编译为动态链接库:
$ gcc -shared -fpic -o bridge.so bridge.c -I<path_to_jdk_include> -L<path_to_jvm>
将上述命令中的<path_to_jdk_include>
和<path_to_jvm>
替换为你的JDK的include目录和JVM所在的目录。最后,我们可以在Golang中调用C函数来间接调用Java方法:
package main
// #cgo CFLAGS: -I<path_to_jvm>/include -I<path_to_jvm>/include/linux
// #cgo LDFLAGS: -L<path_to_jvm>/lib/server -ljvm
// #include <jni.h>
// extern void Java_MyClass_sayHello(JNIEnv *env, jobject obj);
import "C"
import "fmt"
func main() {
C.Java_MyClass_sayHello(nil, nil)
fmt.Println("Hello from Golang!")
}
运行上述代码,你会看到以下输出:
Hello from Java!
Hello from Golang!
除了通过CGO调用JNI之外,我们还可以使用Go的github.com/zgljl2012/go-java
包来直接运行Java代码。这个包提供了一些方便的API,可以帮助我们在Go程序中加载和执行Java类。
首先,我们需要安装github.com/zgljl2012/go-java
包:
$ go get github.com/zgljl2012/go-java
然后,我们可以编写一个Go程序来加载并执行Java类:
package main
import (
"fmt"
"github.com/zgljl2012/go-java"
)
func main() {
jvm, _ := java.CreateJVM(java.CLASSPATH)
defer jvm.DetachCurrentThread()
classLoader, _ := jvm.GetSystemClassLoader()
class, _ := classLoader.LoadClass("MyClass")
method, _ := class.GetMethod("sayHello", "()V")
method.Call(nil)
fmt.Println("Hello from Golang!")
}
运行上述代码,你会看到以下输出:
Hello from Java!
Hello from Golang!
除了使用第三方包之外,我们还可以使用Java虚拟机(JVM)的嵌入式API来加载和执行Java类。这种方法需要手动设置Java虚拟机的类路径和选项。
首先,我们需要引入Java虚拟机(JVM)的头文件:
#include <jni.h>
然后,我们可以编写一个Go程序来加载并执行Java类:
package main
// #cgo CFLAGS: -I<path_to_jvm>/include -I<path_to_jvm>/include/linux
// #cgo LDFLAGS: -L<path_to_jvm>/lib/server -ljvm
// #include <jni.h>
import "C"
import (
"fmt"
"unsafe"
)
func main() {
options := []string{
"-Djava.class.path=<path_to_class>",
}
optionsArray := make([]*C.char, len(options))
for i, s := range options {
optionsArray[i] = C.CString(s)
defer C.free(unsafe.Pointer(optionsArray[i]))
}
C.createJVM((**C.JavaVM)(unsafe.Pointer(&jvm)), (***C.JNIEnv)(unsafe.Pointer(&env)), C.int(len(optionsArray)), &optionsArray[0])
defer jvm.DestroyJavaVM()
classLoader, _ := env.CallObjectMethod(jvm.GetSystemClassLoader(), jvm.LoadClassMethod, env.NewStringUTF("MyClass"))
class, _ := env.CallObjectMethod(classLoader, jvm.LoadClassMethod, env.NewStringUTF("MyClass"))
method, _ := env.CallObjectMethod(class, jvm.GetMethodIDMethod, env.NewStringUTF("sayHello"), env.NewStringUTF("()V"))
env.CallVoidMethod(method, nil)
fmt.Println("Hello from Golang!")
}
//export JNI_OnLoad
func JNI_OnLoad(jvm *C.JavaVM, reserved unsafe.Pointer) C.jint {
return C.JNI_VERSION_1_8
}
var jvm *C.JavaVM
var env *C.JNIEnv
//export createJVM
func createJVM(jvm **C.JavaVM, env ***C.JNIEnv, numOptions C.jint, options **C.char) C.jint {
return C.JNI_CreateJavaVM(jvm, (**C.JNIEnv)(unsafe.Pointer(env)), &(C.JavaVMInitArgs{C.JNI_VERSION_1_8, numOptions, options, C.JNI_FALSE}))
}
将上述代码中的<path_to_jvm>
和<path_to_class>
替换为你的JVM所在的目录和你要加载的Java类的目录。最后,我们可以在Golang中调用C函数来间接调用Java方法:
$ go build -buildmode=c-shared -o libjvm.so && go build main.go
$ LD_LIBRARY_PATH=. ./main
运行上述命令,你会看到以下输出:
Hello from Java!
Hello from G