NDK学习笔记(2)——C与java之间的互相调用

Java调用C

在上一篇博文中,我们完成了一个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

这种方式又被称为回调,即在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这个函数,第一个参数是类实例,第二个参数是方法名,第三个参数是方法签名,至于为什么要使用方法签名和方法名搭配使用,是因为存在这方法重载的因素,这两者搭配便可唯一确定一个方法。
方法签名:(参数类型描述符)返回值类型描述符
类型描述符如下:

特殊字符数据类型特殊说明
Vvoid一般用于表示方法的返回值
Zboolean
Bbyte
Cchar
Sshort
Iint
Jlong
Ffloat
Ddouble
[数组以[开头,配合其他的特殊字符,表示对应数据类型的数组,几个[表示几维数组
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。

demo下载地址

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
JNI(Java Native Interface)是Java提供的一种机制,用于实现Java与其他编程语言(如C++)之间的交互。C++11是C++语言的一个版本,引入了许多新的特性和改进。 在JNI中,可以通过回调函数实现Java与C++之间的双向通信。有返回值的回调函数可以在C++中执行某些操作,并将结果返回给Java。 以下是使用JNI和C++11实现有返回值的回调函数的步骤: 1. 在Java中定义一个接口,用于声明回调函数的方法。例如: ```java public interface Callback { int callbackMethod(); } ``` 2. 在C++中实现该接口,并将其注册到JNI中。例如: ```cpp #include <jni.h> extern "C" { JNIEXPORT void JNICALL Java_com_example_MyClass_registerCallback(JNIEnv* env, jobject obj, jobject callbackObj); JNIEXPORT jint JNICALL Java_com_example_MyClass_nativeMethod(JNIEnv* env, jobject obj); } jobject gCallbackObj; JNIEXPORT void JNICALL Java_com_example_MyClass_registerCallback(JNIEnv* env, jobject obj, jobject callbackObj) { gCallbackObj = env->NewGlobalRef(callbackObj); } JNIEXPORT jint JNICALL Java_com_example_MyClass_nativeMethod(JNIEnv* env, jobject obj) { jclass callbackClass = env->GetObjectClass(gCallbackObj); jmethodID callbackMethod = env->GetMethodID(callbackClass, "callbackMethod", "()I"); jint result = env->CallIntMethod(gCallbackObj, callbackMethod); return result; } ``` 3. 在Java调用C++的回调函数。例如: ```java public class MyClass { static { System.loadLibrary("mylibrary"); } private native void registerCallback(Callback callback); private native int nativeMethod(); public static void main(String[] args) { MyClass myClass = new MyClass(); myClass.registerCallback(new Callback() { @Override public int callbackMethod() { // 执行某些操作并返回结果 return 42; } }); int result = myClass.nativeMethod(); System.out.println("Result: " + result); } } ``` 以上代码演示了如何在Java中注册回调函数,并在C++中调用该回调函数并获取返回值。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值