为Kotlin Android、Rust和Go编译库。

huangapple go评论98阅读模式
英文:

Compile libraries for Kotlin android,rust and go

问题

这个问题是探索性质的,不确定是否适合在Stack Overflow上提问和回答。

背景:

我有一个用Go语言编写的库,我需要编译成多个服务可以使用的形式。

这些服务分别是Kotlin Android、Rust和Go。

我所知道的唯一选项是使用类似SWIG的工具将Go库编译成不同的语言。

问题:

  1. 我不认为SWIG适用于Kotlin。

我正在寻找最佳方法来解决这个问题,以及可以采用的不同方法。

英文:

This question is exploratory in nature, not sure if this fits stack overlflow Q&A.

Context:

I have a library written in golang that I need to compile for multiple services to use.

These services are in Kotlin android,Rust,Golang.

The only option I am aware of is using something like SWIG to compile the go library for different languages.

Problem:

  1. I don't think SWIG works for Kotlin.

I am trying to fish for the best methods to do this and different approaches this can be done.

答案1

得分: 2

对于任何能生成C共享库和头文件的语言,您可以使用SWIG进行封装。同样,对于任何在JVM中运行并能调用Java类的语言,您可以使用SWIG的自动生成的Java绑定。

因此,我们可以按照以下顺序进行一系列操作:

Go -> C -> JNI -> Java -> Kotlin

实际上相当简单。我下面提供了一个示例来展示它的工作原理,因为我之前从未编写过Go或Kotlin。(因此,请对此持保留意见,我可能没有遵循最佳实践!)

此示例假设您已经安装了JDK/JRE、C编译器、Go和kotlinc。

我的demo.go文件如下所示:

  1. package main
  2. import (
  3. "C"
  4. "fmt"
  5. )
  6. //export TestGoFunc
  7. func TestGoFunc(str *C.char) *C.char {
  8. fmt.Printf("Got string: %s\n", C.GoString(str))
  9. return nil
  10. }
  11. func main() {}

而hello.kt文件如下所示:

  1. fun main() {
  2. println("Hello, World!")
  3. test.TestGoFunc("Another string")
  4. }

为了封装它,我编写了以下SWIG接口:

  1. %module test
  2. %{
  3. #include "golib.h"
  4. %}
  5. %include <typemaps.i>
  6. %pragma(java) jniclasscode=%{
  7. static {
  8. System.loadLibrary("test");
  9. }
  10. %}
  11. // 假设您不关心库中的这些内容,静音/整理一些东西
  12. #define _Complex
  13. %ignore _GoString_;
  14. %ignore GoComplex64;
  15. %ignore GoComplex128;
  16. %ignore GoSlice;
  17. %ignore GoInterface;
  18. %include "golib.h"

这是一个针对Java的相当标准的SWIG接口,它隐藏了我们不关心的一些生成的头文件中的内容,并使用一个pragma在Java中自动加载.so文件。

然后,我编写了一个小型的Makefile来构建所有内容,因为构建过程中有很多步骤:

  1. all: libtest.so hello.jar
  2. golib.so: demo.go
  3. go build -o golib.so -buildmode=c-shared demo.go
  4. test_wrap.c: golib.so test.i
  5. swig3.0 -java -Wall test.i
  6. libtest.so: test_wrap.c
  7. gcc -shared -Wall -Wextra test_wrap.c -o libtest.so ./golib.so -I/usr/lib/jvm/default-java/include/ -I/usr/lib/jvm/default-java/include/linux
  8. hello.jar: hello.kt
  9. javac *.java
  10. kotlinc hello.kt -include-runtime -d hello.jar -cp .
  11. jar uvf hello.jar *.class

如果我们构建并运行这个项目,一切都会很好:

  1. $ make
  2. go build -o golib.so -buildmode=c-shared demo.go
  3. swig3.0 -java -Wall test.i
  4. gcc -shared -Wall -Wextra test_wrap.c -o libtest.so ./golib.so -I/usr/lib/jvm/default-java/include/ -I/usr/lib/jvm/default-java/include/linux
  5. javac *.java
  6. kotlinc hello.kt -include-runtime -d hello.jar -cp .
  7. jar uvf hello.jar *.class
  8. adding: test.class(in = 302) (out= 216)(deflated 28%)
  9. adding: testJNI.class(in = 389) (out= 268)(deflated 31%)
  10. $ LD_LIBRARY_PATH=. java -jar hello.jar
  11. Hello, World!
  12. Got string: Another string

我建议在Go中使用-buildmode=c-archive来构建静态库,然后将其链接到SWIG共享对象中,以保持事情更简单。

英文:

For any language that can generate a C shared library and header file you can use SWIG to wrap it. Equally for any language that runs within a JVM and can call Java classes you can make use of SWIG's auto generated Java bindings.

With that we can therefore do a sequence of things that looks like this:

Go -> C -> JNI -> Java -> Kotlin

It's actually fairly sane. I've put together an example below for this to show how it works since I was curious having never written Go nor Kotlin before. (Take this with a pinch of salt therefore, I've probably not hit "best practice" for either!)

This example assumes you have a working JDK/JRE, C compiler, Go installation and kotlinc.

My demo.go looks like this:

  1. package main
  2. import (
  3. &quot;C&quot;
  4. &quot;fmt&quot;
  5. )
  6. //export TestGoFunc
  7. func TestGoFunc(str *C.char) *C.char {
  8. fmt.Printf(&quot;Got string: %s\n&quot;, C.GoString(str))
  9. return nil
  10. }
  11. func main() {}

And hello.kt looks like this:

<!-- language: kotlin -->

  1. fun main() {
  2. println(&quot;Hello, World!&quot;)
  3. test.TestGoFunc(&quot;Another string&quot;)
  4. }

To wrap this I wrote the following SWIG interface:

<!-- language: c -->

  1. %module test
  2. %{
  3. #include &quot;golib.h&quot;
  4. %}
  5. %include &lt;typemaps.i&gt;
  6. %pragma(java) jniclasscode=%{
  7. static {
  8. System.loadLibrary(&quot;test&quot;);
  9. }
  10. %}
  11. // Assuming you don&#39;t care about these in your library silence/neaten stuff
  12. #define _Complex
  13. %ignore _GoString_;
  14. %ignore GoComplex64;
  15. %ignore GoComplex128;
  16. %ignore GoSlice;
  17. %ignore GoInterface;
  18. %include &quot;golib.h&quot;

This is a fairly standard SWIG interface for targeting Java - it hides some of the stuff in the generated header file we don't care about and autoloads the .so file inside Java using a pragma for us.

Then I put together a small Makefile to build everything since there's a bunch of steps to this build:

<!-- language: none -->

  1. all: libtest.so hello.jar
  2. golib.so: demo.go
  3. go build -o golib.so -buildmode=c-shared demo.go
  4. test_wrap.c: golib.so test.i
  5. swig3.0 -java -Wall test.i
  6. libtest.so: test_wrap.c
  7. gcc -shared -Wall -Wextra test_wrap.c -o libtest.so ./golib.so -I/usr/lib/jvm/default-java/include/ -I/usr/lib/jvm/default-java/include/linux
  8. hello.jar: hello.kt
  9. javac *.java
  10. kotlinc hello.kt -include-runtime -d hello.jar -cp .
  11. jar uvf hello.jar *.class

If we build and run this then it all works nicely:

<!-- language: none -->

  1. $ make
  2. go build -o golib.so -buildmode=c-shared demo.go
  3. swig3.0 -java -Wall test.i
  4. gcc -shared -Wall -Wextra test_wrap.c -o libtest.so ./golib.so -I/usr/lib/jvm/default-java/include/ -I/usr/lib/jvm/default-java/include/linux
  5. javac *.java
  6. kotlinc hello.kt -include-runtime -d hello.jar -cp .
  7. jar uvf hello.jar *.class
  8. adding: test.class(in = 302) (out= 216)(deflated 28%)
  9. adding: testJNI.class(in = 389) (out= 268)(deflated 31%)
  10. $ LD_LIBRARY_PATH=. java -jar hello.jar
  11. Hello, World!
  12. Got string: Another string

I'd be tempted to use -buildmode=c-archive for Go to build a static library and then link that into the SWIG shared object instead just to keep things simpler in that regards though.

huangapple
  • 本文由 发表于 2022年6月1日 23:19:08
  • 转载请务必保留本文链接:https://go.coder-hub.com/72464343.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定