在上一篇博文中,我们完成了一个NDK项目的基本配置以及各个文件的功能,现在来写一下其他功能,demo下载在文章末尾。
老样子,在MainActivity写一个native方法:
public native String stringTransfer(String s);
然后鼠标放在方法上按住alt+enter,自动添加了C的相关函数,然后我们改改:
//需要加这句,表明以下代码是用C编译而不是c++,否则在编译时会报错
extern "C"
JNIEXPORT jstring JNICALL
Java_com_ndkdemo_ustc_jnitest_MainActivity_stringTransfer(JNIEnv *env, jobject instance,
jstring s_) {
//获取字符串
const char *s = env->GetStringUTFChars(s_, 0);
char *temp = (char *) s;
if (s == NULL)
return NULL;
char *fromC = (char *) "add I am from C";
//获取字符串长度
env->GetStringLength(s_);
//字符串拼接
strcat(temp, fromC);
//释放字符串所占的内存空间
env->ReleaseStringUTFChars(s_, s);
return env->NewStringUTF(temp);
}
获取字符串的函数根据编码方式的不同可以分为两种:UTF-8和unicode - unicode GetStringChars / ReleaseStringChars:获取/释放字符串 GetStringLength:获取字符串长度 - UTF-8 GetStringUTFChars / ReleaseStringUTFChars:获取/释放字符串 GetStringUTFLength:获取字符串长度 除此之外还有一些其他方法: GetStringCritical /ReleaseStringCritical :看到critical就知道这玩意八成是为了防止死锁,获得/释放一个Unicode格式的字符串指针,可能返回一个字符串的副本(在该函数对区间内,不能使用任何JNI函数),此函数可以阻止GC回收。 GetStringRegion / GetStringUTFRegion:把字符串复制到一个预先分配的缓冲区内,会做越界检查,不做任何内存分配,不会抛出内存溢出异常。
java代码:
public native int[] addElement(int[] array);
C代码:
extern "C"
JNIEXPORT jintArray JNICALL
Java_com_ndkdemo_ustc_jnitest_MainActivity_addElement(JNIEnv *env, jobject instance,
jintArray array_) {
//得到数组元素
jint *array = env->GetIntArrayElements(array_, NULL);
//得到数组长度
jsize size = (*env).GetArrayLength(array_);
//创建一个新的数组
jintArray res = env->NewIntArray(size);
//遍历数组
for (int i = 0; i < size; i++) {
*(array + i) += 10;
}
//把值拷贝到res数组中,不可以直接返回array
env->SetIntArrayRegion(res, 0, size, array);
//释放数组所占内存
env->ReleaseIntArrayElements(array_, array, 0);
//返回
return res;
}
相关方法: 构造新数组的方式大同小异:NewIntArray(),NewCharArray()等,参数是数组的长度。 GetXXXArrayElements():获得某一类型的数组元素,返回的是数组的首地址。 GetArrayLength():获得数组长度 SetXXXArrayRegion():设置某个数组的元素,参数分别是被设置的数组,原数组的其实位置,原数组的结束位置,原数组。 ReleaseXXXArrayElements():释放某个数组。
这种方式又被称为回调,即在C代码里通过反射的方式获取java的类的字节码,然后再获取对应的方法进行调用。 java代码:
public native void callBackAdd();
回调的java方法:
public void addCallBack(int x,int y){
addCallBackTx.setText("和为"+(x+y));
}
C代码:
extern "C"
JNIEXPORT void JNICALL
Java_com_ndkdemo_ustc_jnitest_MainActivity_callBackAdd(JNIEnv *env, jobject instance) {
//得到字节码
jclass jclazz = env->GetObjectClass(instance);
//得到方法
//最后一个参数是方法签名:(参数类型描述符)返回值类型描述符
jmethodID methodId = env->GetMethodID(jclazz, "addCallBack", "(II)V");
//调用方法
env->CallVoidMethod(instance, methodId, 5, 10);
}
这里主要介绍GetMethodID这个函数,第一个参数是类实例,第二个参数是方法名,第三个参数是方法签名,至于为什么要使用方法签名和方法名搭配使用,是因为存在这方法重载的因素,这两者搭配便可唯一确定一个方法。 方法签名:(参数类型描述符)返回值类型描述符 类型描述符如下:
特殊字符 | 数据类型 | 特殊说明 |
---|---|---|
V | void | 一般用于表示方法的返回值 |
Z | boolean | |
B | byte | |
C | char | |
S | short | |
I | int | |
J | long | |
F | float | |
D | double | |
[ | 数组 | 以[开头,配合其他的特殊字符,表示对应数据类型的数组,几个[表示几维数组 |
L | 全类名 | 引用类型 以L开头、;结尾,中间是引用类型的全类名 |
比如:
方法 | 签名 |
---|---|
public void test1(){} | ()V |
public void test2(String str) | (Ljava/lang/String;)V |
public int test3(){} | ()I |
public void addCallBack(int x,int y) | (II)V |
其实回调的一个重要作用就在于可以在回调方法里更新UI,比如上例中就设置了TextView。