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

特殊字符

数据类型

特殊说明

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。

demo下载地址

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏微信公众号:Java团长

JVM内幕:Java虚拟机详解

这篇文章解释了Java 虚拟机(JVM)的内部架构。下图显示了遵守 Java SE 7 规范的典型的 JVM 核心内部组件。

782
来自专栏玄魂工作室

Python黑帽编程2.2 数值类型

Python黑帽编程2.2 数值类型 数值类型,说白了就是处理各种各样的数字,Python中的数值类型包括整型、长整型、布尔、双精度浮点、十进制浮点和复数,这...

2719
来自专栏技术/开源

一道javascript面试题

下面表达式比较的结果分别是什么? 1. []=="0" 2. []==0 3. "0"==0 4. []==false 5. []==[] 大家可以...

20210
来自专栏Golang语言社区

Golang语言之异常处理

在编写Go语言代码的时候,我们应该习惯使用error类型值来表明非正常的状态。作为惯用法,在Go语言标准库代码包中的很多函数和方法也会以返回error类型值来表...

33313
来自专栏用户2442861的专栏

java类加载过程

转载请注明出处:http://blog.csdn.net/ns_code/article/details/17881581

391
来自专栏专注数据中心高性能网络技术研发

[Effective Modern C++(11&14)]Chapter 1: Deducing Types

1523
来自专栏用户2442861的专栏

【深入Java虚拟机】之四:类加载机制

转载请注明出处:http://blog.csdn.net/ns_code/article/details/17881581

421
来自专栏JavaQ

高并发编程-synchronized深入解析

对于单一JVM来说,synchronized可以保证在并发情况下,同一时刻只有一个线程执行某个方法或某段代码。synchronized可用于修饰普通方法、静态方...

784
来自专栏我的小碗汤

利用golang优雅的实现单实例

平时编写代码过程中,经常会遇到对于全局角度只需运行一次的代码,比如全局初始化操作,设计模式中的单例模式。针对单例模式,java中又出现了饿汉模式、懒汉模式,再配...

993
来自专栏wOw的Android小站

[设计模式]之十二:状态模式

在很多情况下,一个对象的行为取决于一个或多个动态变化的属性,这样的属性叫做状态,这样的对象叫做有状态的(stateful)对象,这样的对象状态是从事先定义好的一...

441

扫码关注云+社区