java union jni,Android JNI开发基础学习

JNI简述

JNI是Java Native Interface的缩写,是Java调用C++的规范。用于Java与本地C、C++交互

关于环境搭建,这里不再叙述,具体可参考百度搜索结果。

基础示例

java中定义native方法:

public native String stringFromJNI();

native library 中C++实现:

#include

#include

extern "C" JNIEXPORT jstring JNICALL

Java_fun_qianxiao_passwordmanager_MainActivity_stringFromJNI(

JNIEnv* env,

jobject jobj) {

std::string hello = "Hello from C++";

return env->NewStringUTF(hello.c_str());

}

可以看到一个java中无参方法在c++中实现,函数名的变化规则,同时多了2个参数(JNIEnv* env、jobject jobj)。

JNIEnv* 代表了Java环境,通过这个JNIEnv* 指针,就可以对Java端的代码进行操作。

jobject 代表这个native方法的类实例或这个类的class对象。

Java类型在C/C++中的对应关系

Java类型

本地类型

Jni别名

int

long

jint/jsize

long

__int64

jlong

byte

signed char

jbyte

boolean

unsigned char

jbollean

char

unsigned short

jchar

short

short

jshort

float

float

jfloat

double

double

jdouble

Object

_jobject*

jobject

类型签名

我的理解,这个同smali语法中类型对应。

Java类型

签名符号

boolean

Z

byte

B

char

C

short

S

int

I

long

J

float

F

double

D

Object

L+路径名+;(如String对应Ljava/long/String;)

某类型数组

[+类型符号(如int[]对应[I)

C++访问Java

访问修改类变量

如下,java MainActivity 类中有一int类型的属性i:

public class MainActivity {

public int i = 0;

}

C++中对变量i进行获取:

//首先通过JNIEnv指针获取jobject中对象的class对象

jclass clazz = env->GetObjectClass(obj);

//通过JNIEnv指针获取Java中名为i的int型变量

jfieldID id_i = env->GetFieldID(clazz,"i","I");

//通过JNIEnv指针获取变量的值

jint i = env->GetIntField(obj,id_i);

对变量进行修改:

//通过JNIEnv指针修改i的值为100,注意jint对应C++是long,所以后面要L

env->SetIntField(obj,id_number,100L);

同理使用GetField和SetField对其他类型变量进行获取和修改。

C++调用Java方法

一般调用方式

调用方式1( 最常用的方式 ):

CallMethod(jobject obj,jmethodID id,….);

表示方法返回类型。假设Java中有如下方法:String fun1(String s,int a,char c,double d);则在C++中如何调用?

//首先通过JNIEnv指针获取jobject中对象的class对象

jclass clazz = env->GetObjectClass(obj);

//通过JNIEnv指针获取Java中的fun1方法的方法id(注意参数3)

jmethodID id_fun1 = env->GetMethodID(clazz,"fun1","(Ljava/long/String;ICD)Ljava/long/String;");

//通过JNIEnv指针调用fun1方法

jstring doubles = (jstring)env->CallObjectMethod(obj,id_fun1,env->NewStringUTF("i am string"),6L,L'A',1.2);

调用方式2( va_list 指向参数表,很少使用 ):

CallMethod(jobject obj,jmethodID id,va_list lst);

调用方式3( jvalue 看理解为指向数组的指针 ):

CallMethod(jobject obj,jmethodID id,jvalue* v);

jvalue 是一个union联合体,内有jbollean、jbyte、jchar、jshort、jint、jlong、jfloat、jdouble、jobject类型,给其中一个类型赋值之后,这个union就是这种类型了。

如Java中有如下方法 boolean fun2(int a,double b,char c) ;则在C++中调用如下:

//首先通过JNIEnv指针获取jobject中对象的class对象

jclass clazz = env->GetObjectClass(obj);

//通过JNIEnv指针获取Java中的fun1方法的方法id(注意参数3)

jmethodID id_fun2 = env->GetMethodID(clazz,"fun2","(IDC)Z");

jvalue* args = new jvalue[3];//定义jvalue数组

args[0].i = 10L;//i表示jvalue中的jint

args[1].d = 3.44;//d表示jvalue中的jdouble

args[2].c = L'a';//c表示jvalue中的jchar

jboolean result = env->CallBooleanMethod(obj, id_fun2 , args);

delete[] args;

其他调用方式:

CallStaticMethod调用静态方法, CallNonvirtualMethod 调用不虚方法(父类方法)。

调用复杂类型

以下面这个示例说明:

如Java中有BaseClass类、其子类ChildClass类,MainActivity类中有一BaseClass 引用指向ChildClass 的对象:

public class BaseClass {

public void fun(){

System.out.println("BaseClass:fun");

}

}

public class ChildClass extends BaseClass {

@Override

public void fun() {

System.out.println("ChildClass:fun");

}

}

public class MainActivity {

public BaseClass baseClass = new ChildClass();

}

我们怎么使用C++调用BaseClass 或是ChildClass类中的方法呢?

//首先通过JNIEnv指针获取jobject中对象的class对象

jclass clazz = env->GetObjectClass(obj);

//通过JNIEnv指针获取Java中名为baseClass的BaseClass型对象

jfieldID id_baseClass = env->GetFieldID(clazz,"baseClass","Lfun/qianxiao/jnidemo/BaseClass;");

//通过JNIEnv指针获取baseClass的object对象

jobject baseClass = env->GetObjectField(obj,id_baseClass);

//通过JNIEnv指针获取BaseClass的class对象

jclass clazz_baseClass = env->FindClass("fun/qianxiao/jnidemo/BaseClass");

//通过JNIEnv指针获取BaseClass中的fun方法的id

jmethodID id_baseClass_fun = env->GetMethodID(clazz_baseClass,"fun","()V");

//调用fun方法,实际执行子类的方法(多态)

env->CallVoidMethod(baseClass,id_baseClass_fun);

//调用父类中的fun方法

env->CallNonvirtualVoidMethod(baseClass,clazz_baseClass,id_baseClass_fun);

参考:https://blog.csdn.net/yuanzhihua126/article/details/78992068

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值