Java/Kotlin和C/C++的混编

Java和C/C++混编技术是一种常用的开发技术,目的是使用C/C++弥补Java在高性能计算方面的不足。如Android的Native开发,我们经常是使用Native代码来处理一些复杂的计算,比如图象处理。实现Java/Kotlin和C/C++通讯的接口叫Jni,下面我们来看一下Jni接口的基本用法。

1. Java/Kotlin调用C/C++方法

首先在Java/Kotlin中声明一个Native的空方法

package com.test.native;

public class Main {
  //Java: 使用native关键字声明
  public native String getString(int index);

  //Kotlin: 使用external关键字声明
  public external fun getString(index: Int): String
}

然后,在使用这个方法之前,如果你的C/C++库是动态的so库,那么必须提前在静态代码块中加载Native库,例如:

//Java: 在类的静态代码块中加载
static {
  System.loadlibrary("native-lib");
}

//Kotlin: 在伴生对象的初始化代码块中加载
companion object {
  init {
    System.loadlibrary("native-lib")
  }
}

参数名表示库的名称,这个"native-lib"完整的库文件名称是“libnative-lib.so”(为啥叫这个?这是Android Studio自动生成的,大家都懒得改名),libxxx.so中间这个xxx就是动态库的名称,这是C/C++通用的命名方式。

注意,一般情况下我们的程序一定有一个动态库作为主库,主库可以链接其他的库,然后Java/Kotlin代码中通过加载这个动态的主库来实现调用。

之后,我们需要在Native代码中写一个对应的Jni方法。在我们的native-lib.cpp中:

extern "C" {
JNIEXPORT jstring
JNICALL Java_com_test_native_Main_getString(JNIEnv * env, jobject obj, jint index) {
  return env->NewStringUTF(sprintf("from native index: %d", index));
}
}

这个方法有一系列的修饰符和一个很长很长的方法名,JNIEXPORT后边是返回值类型,JNICALL后边是方法的全名,名称第一个单词是Java(kotlin也得写Java,而不是用Kotlin开头),然后跟完整的包名和Java/Kotlin方法名,参数列表的前两个是默认传递的Java参数,包括Java环境和当前类的实例,之后才是真正需要的参数列表。Java中的类型都被映射成了jstring、jint等等,部分可以直接强转成C/C++类型,如jint,但是jstring不行,必须通过JNIEnv操作转换才能确保正确。

Java/Kotlin调用C/C++就这么简单几步,然后你就可以在JNI的方法里写你想用的功能了。

2. C/C++调用Java/Kotlin方法、属性

(1) 方法

这个相对麻烦一点,因为方法分了静态和非静态、有返回值和无返回值等等。

首先,有一个Java/Kotlin类,test.java:

package com.test.native;

class Main {
  int index;
  static String name;
  public String getString(int index );
  public static String getName();
}

然后再C/C++中调用这个方法:

jclass Main = env->FindClass("com/test/native/Main");
jmethodID main_constructor = env->GetMethodID(Main, "<init>", "()V");
jobject main = env->NewObject(Main, main_constructor);
jmethodID method = env->GetMethodID(Main, "getString", "(I)Ljava/lang/String;");
int index = 1;
env->CallObjectMethod(main, method, index);

主要过程分为三步:1,映射Main这个类,并且新建对象(也可以从Java传过来);2,找到需要调用的方法 getString;3,调用方法。

其中,类的分隔符使用 /,构造方法使用 <init>来代替。在获取方法的时候,最后一个参数叫做方法的签名,如 (I)Ljava/lang/String;。括号内是方法的参数,括号后边是方法的返回类型,I代表int类型, Ljava/lang/String代表String类型。完整的类型对应表:

类型 缩写
short S
int I
long J
float F
double D
boolean Z
byte B
char C
void V
对象类型 L包名,如:Ljava/lang/String
数组类型 前边加 [ 修饰符, 如: [S

对象参数或返回值后边必须要加分号,基本类型不用加,如 "(Ljava/lang/String;I)V","(IB)Ljava/lang/String;"。

静态方法调用

jclass Main = env->FindClass("com/test/native/Main");
jmethodID static_method = env->GetStaticMethodID(Main, "getName", "()Ljava/lang/String;");
env->CallStaticObjectMethod(Main, static_method);

流程类似,只是少了创建对象的过程。

(2) 属性

属性和方法比较类似,只是多了个修改的步骤:

jclass Main = env->FindClass("com/test/native/Main");
jmethodID main_constructor = env->GetMethodID(Main, "<init>", "()V");
//属性
jfieldID field = env->GetFieldID(Main, "index", "I");
jint index = env->GetIntField(main, field);
//更改属性值
env->SetIntField(main, field, 2);

//静态属性
jfieldID static_field = env->GetStaticFieldID(Main, "name", "Ljava/lang/String");
jobject name = env->GetStaticObjectField(Main, static_field);
//更改静态属性值
env->SetStaticObjectField(Main, static_field, env->NewStringUTF("main"));

代码看着多了点,其实原理并不复杂。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,445评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,889评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,047评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,760评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,745评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,638评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,011评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,669评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,923评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,655评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,740评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,406评论 4 320
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,995评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,961评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,023评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,483评论 2 342

推荐阅读更多精彩内容