脚本之家,脚本语言编程技术及教程分享平台!
分类导航

Python|VBS|Ruby|Lua|perl|VBA|Golang|PowerShell|Erlang|autoit|Dos|bat|

服务器之家 - 脚本之家 - Golang - Golang开发动态库的实现

Golang开发动态库的实现

2020-05-31 11:32键盘上的麒麟臂 Golang

这篇文章主要介绍了Golang开发动态库的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

我们平时使用的动态库都是由C/C++开发最后生成的.so文件。

可以先看看一个JNI的开发过程。

一. 开发JNI

有两种方式,现在一种比较快的方式是AndroidStudio你在创建项目选择Module的时候它会给你个JNI的模板,直接使用那个就行。

但是我还是比较喜欢传统的方法。

简单来说传统的方式就是你用命令来把java文件变成C++的头文件

简单演示一遍,先写个java类

?
1
2
3
4
5
6
7
8
9
public class TestJni {
 
  static {
    System.loadLibrary("KylimTest");
  }
 
  public static native String getMsg();
 
}

定义了一个native修饰的方法,在代码调用这个方法之后JNI就会自动调用到动态库中相应的方法。

将这个类用命令生成头文件,来到文件夹路径下输入命令

?
1
javah -jni 包名.类名

可以看到默认会生成一个.h的头文件,自动命名为 包名_类名.h

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class com_kylim_nativetest_TestJni */
 
#ifndef _Included_com_kylim_nativetest_TestJni
#define _Included_com_kylim_nativetest_TestJni
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:   com_kylim_nativetest_TestJni
 * Method:  getMsg
 * Signature: (I)Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_kylim_nativetest_TestJni_getMsg
 (JNIEnv *, jclass);
 
#ifdef __cplusplus
}
#endif
#endif

主要的核心就是这句

?
1
2
JNIEXPORT jstring JNICALL Java_com_kylim_nativetest_TestJni_getMsg
 (JNIEnv *, jclass);

其它的我也不清楚,都是C相关的, 如果你嫌用命令生成麻烦,你可以自己创建一个.h文件然后方法命名就按照这样的规范去写

头文件只是为了定义,我们需要自己写原文件,所以要创建一个.cpp结尾的文件

?
1
2
3
4
5
6
7
8
#include "com_kylim_nativetest_TestJni.h"
 
JNIEXPORT jstring JNICALL Java_com_kylim_nativetest_TestJni_getMsg
    (JNIEnv *env, jclass cls){
 
  jstring result = env->NewStringUTF("结果是");
  return result;
}

方法命名是有规范的,看Demo也知道怎么规范了,没必要多解释,这样两端的代码就写完了,但是仅仅这样是无法运行项目的。

还需要些一些配置,因为在AndroidStudio中是Gradle去帮我们编译C++的代码,所以需要写这些配置。如果你不是用AS开发,你用其它工具开发直接生成.so文件再丢进AS中的话,可以忽略这一步。

先看看我的Jni目录

Golang开发动态库的实现

要创建一个Android.mk

?
1
2
3
4
5
6
7
8
9
#固定写法
LOCAL_PATH:=$(call my-dir)
#固定写法
include $(CLEAR_VARS)
#生成so名称
LOCAL_MODULE := KylimTest
LOCAL_SRC_FILES := testone.cpp
#固定写法
include $(BUILD_SHARED_LIBRARY)

具体的配置可自行去查找,这里不是主要讲JNI的,所以就不讲这么细。

还需要一个Application.mk

?
1
2
3
4
5
6
# 选择不同的 ABI,多个使用空格作为分隔符,全部是all
# APP_ABI := armeabi armeabi-v7a
APP_ABI := armeabi-v7a
 
# 指定要使用的运行时
APP_STL := c++_static

当然这样还不行,都说了是Gradle进行编译,那么肯定还要在Gradle中写一些配置

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
android {
   defaultConfig {
        ndkBuild {
        //指定 Application.mk 的路径
        arguments "NDK_APPLICATION_MK:=src/main/jni/Application.mk"
        //指定生成哪些平台的 so 文件
        abiFilters "armeabi-v7a"
        //cFlags 和 cppFlags 是用来设置环境变量的, 一般不需要动
        cFlags "-DTEST_C_FLAG1", "-DTEST_C_FLAG2"
        cppFlags "-DTEST_CPP_FLAG2", "-DTEST_CPP_FLAG2"
      }
   }
 
  sourceSets { main { jni.srcDirs = ['src/main/jni'] } }
 
  externalNativeBuild {
    ndkBuild {
      path file('src/main/jni/Android.mk')
    }
  }
 
}

这样就能简单的跑一个JNI的Demo,总的来说就是Java这边写一个类定义一些native方法和加载,C++这边写具体的方法实现。

二.用Go开发动态库

上面说的原生方法是用C/C++进行开发的,那么如果你不会C++的话怎么办,C++的学习也并非这么容易,就拿两边的类型来说,一开始新手肯定会碰到类型转换的问题,往往会先劝退一些人,但是Go不一样,有Java基础的话学起go还是挺快的。

那么用Go开发的动态库是怎样的?也是SO文件吗,是不是也像C++一样,编译后经过某步操作生成SO文件。

我看到网上有些文章确实是写怎么生成so的,但是说得太少,感觉不可靠,直到我看到官方有写。

Golang开发动态库的实现

可以在官方中看到是有一个mobile的库的

https://github.com/golang/mobile

读下去它会指引你去wiki

https://github.com/golang/go/wiki/Mobile

Golang开发动态库的实现

可以看出它会打出一个aar的文件,那么aar对于我们接入来说确实很方法,但我很想探究这个aar里面究竟是什么,所以我们需要打出一个aar然后解压看看它里面到底是什么

这里先说一下,下载这个库之前,你本地肯定先要配置好Go的环境

然后按照这里的流程就行下载

?
1
go get -d golang.org/x/mobile/example/basic

但这输入这条命令需要科学上网的方式才能下载,总的来说很麻烦。

所以我们可以直接克隆mobile的库,就是上面的这个链接 https://github.com/golang/mobile

直接下载下来,除此之外,还需要tools,这些都在Go中,链接 https://github.com/golang/tools

将这两个下载下来,然后拷贝到你的Go的以下路径

go/src 创建一个文件夹golang.org/x ,把这两个文件夹丢进去

然后输入命令

?
1
gomobile init

可以输入命令 查看安装配置是否成功

?
1
gomobile bind -help

如果配置成功会给你一些提示

Golang开发动态库的实现

我感觉文档写得还是不算清楚,但是它有告诉你用什么命令生成aar

?
1
gomobile bind -o app/hello.aar -target=android golang.org/x/mobile/example/bind/hello

你在GoPath中创建一个Go文件,我是用GoLand进行开发的,项目的目录设置成GoPath,编写完之后,可以直接在文件中运行

?
1
gomobile bind -o 输入文件名.aar -target=android

这样就能在文件夹中生成一个aar

接着我们看看aar里面是什么,解压

Golang开发动态库的实现

首先可以看到生成这些ABI的so文件,再看看Manifest

Golang开发动态库的实现

这里有限制最低版本,所以如果你的版本比他还低的话就需要注意一下了

然后res里面是可以看到没有文件的。

最后我们反编译class文件

Golang开发动态库的实现

因为这不是Demo,直接是写公司的项目,所以有些地方要码,但是不影响。

从这里看得出,go帮我们生成了一个java文件,这个java文件定义了再Go中命名的原生方法。

其实从这里就可以看出,Go用的也基本是我们最上面写的JNI的方法,只是他帮你封装起来了而已

但是他的原生代码是不是转成C++的我就不清楚了,因为我不会反编译SO文件。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。

原文链接:https://www.jianshu.com/p/351fe9a46df4

延伸 · 阅读

精彩推荐
  • Golanggolang json.Marshal 特殊html字符被转义的解决方法

    golang json.Marshal 特殊html字符被转义的解决方法

    今天小编就为大家分享一篇golang json.Marshal 特殊html字符被转义的解决方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧 ...

    李浩的life12792020-05-27
  • Golanggolang 通过ssh代理连接mysql的操作

    golang 通过ssh代理连接mysql的操作

    这篇文章主要介绍了golang 通过ssh代理连接mysql的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...

    a165861639710342021-03-08
  • GolangGolang中Bit数组的实现方式

    Golang中Bit数组的实现方式

    这篇文章主要介绍了Golang中Bit数组的实现方式,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...

    天易独尊11682021-06-09
  • GolangGolang通脉之数据类型详情

    Golang通脉之数据类型详情

    这篇文章主要介绍了Golang通脉之数据类型,在编程语言中标识符就是定义的具有某种意义的词,比如变量名、常量名、函数名等等,Go语言中标识符允许由...

    4272021-11-24
  • Golanggolang的httpserver优雅重启方法详解

    golang的httpserver优雅重启方法详解

    这篇文章主要给大家介绍了关于golang的httpserver优雅重启的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,...

    helight2992020-05-14
  • Golanggolang如何使用struct的tag属性的详细介绍

    golang如何使用struct的tag属性的详细介绍

    这篇文章主要介绍了golang如何使用struct的tag属性的详细介绍,从例子说起,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看...

    Go语言中文网11352020-05-21
  • Golanggo日志系统logrus显示文件和行号的操作

    go日志系统logrus显示文件和行号的操作

    这篇文章主要介绍了go日志系统logrus显示文件和行号的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...

    SmallQinYan12302021-02-02
  • Golanggo语言制作端口扫描器

    go语言制作端口扫描器

    本文给大家分享的是使用go语言编写的TCP端口扫描器,可以选择IP范围,扫描的端口,以及多线程,有需要的小伙伴可以参考下。 ...

    脚本之家3642020-04-25