Android之JNI开发入门

在jni目录下创建 Application.mk 在里面指定

APP_ABI := armeabi x86

APP_PLATFORM := android-14

  • javah 命令:生成java代码中本地方法名对应的C语言的函数名

使用方法:javah com.xfhy.jnihelloworld.MainActivity

  • jdk 1.7 项目 src目录下运行javah

  • jdk 1.6 项目 bin目录下 classes文件夹

  • javah native方法声明的java类的全类名

5.jni简便开发流程

===========

  • ① 写java代码 native 声明本地方法

  • ② 添加本地支持 右键单击项目->andorid tools->add native surport

  • 如果发现 finish不能点击需要给工作空间配置ndk目录的位置

  • window->preferences->左侧选择android->ndk 把ndk解压的目录指定进来

  • ③ 如果写的是.c的文件 先修改一下生成的.cpp文件的扩展名 不要忘了 相应修改Android.mk文件中LOCAL_SRC_FILES的值

  • ④ javah生成头文件 在生成的头文件中拷贝c的函数名到.c的文件

  • ⑤ 解决CDT插件报错的问题

  • 右键单击项目选择 properties 选测 c/c++ general->paths and symbols->include选项卡下->点击add…->file system 选择ndk目录下 platforms文件夹 对应平台下(项目支持的最小版本)

usr 目录下 arch-arm -> include 确定后 会解决代码提示和报错的问题

  • ⑥编写C函数 如果需要单独编译一下c代码就在c/c++视图中找到小锤子,点一下小锤子就编译了

  • 如果想直接运行到模拟器上 就不用锤子了,直接右键run as,然后就自动编译了

  • ⑦ java代码中不要忘了 System.loadlibrary();

static {

System.loadLibrary(“hello”); //加载动态链接库

}

6. Java向C传递一些基本的类型,处理

======================

将一个jstring转换成一个c语言的char* 类型工具方法

char* _JString2CStr(JNIEnv* env, jstring jstr) {

char* rtn = NULL;

jclass clsstring = (*env)->FindClass(env, “java/lang/String”);

jstring strencode = (*env)->NewStringUTF(env,“GB2312”);

jmethodID mid = (*env)->GetMethodID(env, clsstring, “getBytes”, “(Ljava/lang/String;)[B”);

jbyteArray barr = (jbyteArray)(*env)->CallObjectMethod(env, jstr, mid, strencode); // String .getByte(“GB2312”);

jsize alen = (*env)->GetArrayLength(env, barr);

jbyte* ba = (*env)->GetByteArrayElements(env, barr, JNI_FALSE);

if(alen > 0) {

rtn = (char*)malloc(alen+1); //“\0”

memcpy(rtn, ba, alen);

rtn[alen]=0;

}

(*env)->ReleaseByteArrayElements(env, barr, ba,0);

return rtn;

}

int类型

JNIEXPORT jint JNICALL Java_com_xfhy_javapassdata_JNI_add

(JNIEnv * env, jobject clazz, jint x, jint y){

return x+y; //直接返回x+y

}

String类型

JNIEXPORT jstring JNICALL Java_com_xfhy_javapassdata_JNI_sayHelloInC

(JNIEnv *env, jobject clazz, jstring str){

//将jstring转换成char* 类型

char* cstr = _JString2CStr(env,str);

//调用C语言的strlen测量cstr字符串的长度

int length = strlen(cstr);

int i=0;

for(i=0; i<length; i++){

*(cstr+i) += 1; //将字符串+1

}

return (env)->NewStringUTF(env,cstr); //将char 类型转换成String类型返回

}

int[]类型

JNIEXPORT jintArray JNICALL Java_com_xfhy_javapassdata_JNI_arrElementsIncrease

(JNIEnv *env, jobject clazz, jintArray jArray) {

//jsize (GetArrayLength)(JNIEnv, jarray); 返回数组长度

int length = (*env)->GetArrayLength(env,jArray);

//jint* (GetIntArrayElements)(JNIEnv, jintArray, jboolean*); 最后一个参数表示是否拷贝,可以不用传值

//返回int* 返回该数组的首地址 这样就可以直接通过该指针直接操作该数组了

int* cArray = (*env)->GetIntArrayElements(env,jArray,NULL);

int i;

for(i=0; i<length; i++) {

*(cArray+i) += 10;

}

return jArray; //直接将原数组返回(这时已经是修改过了的)

}

7.C代码中向logcat输出内容

=================

1.Android.mk文件增加以下内容

LOCAL_LDLIBS += -llog

2.C代码中增加以下内容

#include <android/log.h>

#define LOG_TAG “xfhy”

#define LOGD(…) __android_log_print(ANDROID_LOG_DEBUG, LOG_TAG, VA_ARGS)

#define LOGI(…) __android_log_print(ANDROID_LOG_INFO, LOG_TAG, VA_ARGS)

  • define C的宏定义 起别名 #define LOG_TAG “xfhy” 给”xfhy”起别名LOG_TAG

  • #define LOGI(…) android_log_print(ANDROID_LOG_INFO, LOG_TAG, __VA_ARGS),ANDROID_LOG_DEBUG表示优先级 debug ANDROID_LOG_INFO表示info 这些在log.h中可以看到

  • 给 __android_log_print函数起别名 写死了前两个参数 第一个参数 优先级 第二个参数TAG

  • VA_ARGS:是可变参数的固定写法

  • LOGI(…)在调用的时候 用法跟printf()一样

8. C代码回调java方法

===============

首先需要了解:

Java反射

public class Demo {

public static void main(String[] args) {

//1.获取字节码对象

Class util = Utils.class;

try {

//2.获取Method对象 方法名,参数类型

Method method = util.getMethod(“test”, String.class);

//3.通过字节码对象创建一个Obejct

Object obj = util.newInstance();

//4.通过对象调用方法 对带有指定参数的指定对象调用由此 Method 对象表示的底层方法

method.invoke(obj, “hello”);

} catch (Exception e) {

e.printStackTrace();

}

}

public class Utils {

public void test(String string) {

System.out.println(string);

}

}

}

C代码回调java方法

  • ① 找到字节码对象

  • //jclass (FindClass)(JNIEnv, const char*);

  • //第二个参数 要回调的java方法所在的类的路径 “com/itheima/callbackjava/JNI”

  • ② 通过字节码对象找到方法对象

  • //jmethodID (GetMethodID)(JNIEnv, jclass, const char*, const char*);

  • 第二个参数 字节码对象 第三个参数 要反射调用的java方法名 第四个参数 要反射调用的java方法签名

  • javap -s 要获取方法签名的类的全类名 项目/bin/classes 运行javap

  • ③ 通过字节码创建 java对象(可选) 如果本地方法和要回调的java方法在同一个类里,可以直接用 jni传过来的java对象 调用创建的Method

  • jobject obj =(*env)->AllocObject(env,claz);

  • 当回调的方法跟本地方法不在一个类里 需要通过刚创建的字节码对象手动创建一个java对象

  • 再通过这个对象来回调java的方法

  • 需要注意的是 如果创建的是一个activity对象 回调的方法还包含上下文 这个方法行不通!!!回报空指针异常

  • ④ 反射调用java方法

  • //void (CallVoidMethod)(JNIEnv, jobject, jmethodID, …);

  • 第二个参数 调用java方法的对象 第三个参数 要调用的jmethodID对象 可选的参数 调用方法时接收的参数

/*

  • Class: com_xfhy_callbackjava_JNI java里的类的完整路径

  • Method: callbackvoidmethod java里面的方法名

  • Signature: ()V 方法签名

*/

JNIEXPORT void JNICALL Java_com_xfhy_callbackjava_JNI_callbackvoidmethod

(JNIEnv *env, jobject clazz) {

//1.找到字节码对象

//jclass (FindClass)(JNIEnv, const char*);

//参数: env,需要反射的对象的全路径

jclass claz = (*env)->FindClass(env,“com/xfhy/callbackjava/JNI”);

//2.通过字节码对象找到方法对象

//jmethodID (GetMethodID)(JNIEnv, jclass, const char*, const char*);

//第二个参数 字节码对象 第三个参数 要反射调用的java方法名 第四个参数 要反射调用的java方法签名

//javap -s 要获取方法签名的类的全类名 项目/bin/classes 运行javap

jmethodID methodID = (*env)->GetMethodID(env,claz,“helloFromJava”,“()V”);

//3.通过字节码创建java对象(可选),如果本地方法和要调用的java方法在同一个类里,可以直接用jni传过来的java对象,调用创建的Method

//jobject obj =(*env)->AllocObject(env,claz);

//* 当回调的方法跟本地方法不在一个类里 需要通过刚创建的字节码对象手动创建一个java对象

//* 再通过这个对象来回调java的方法

//* 需要注意的是, 如果创建的是一个activity对象, 回调的方法还包含上下文 ,这个方法行不通!!!回报空指针异常

//4.反射调用java方法

//void (CallVoidMethod)(JNIEnv, jobject, jmethodID, …);

//第二个参数 调用java方法的对象 第三个参数 要调用的jmethodID对象 可选的参数 调用方法时接收的参数

(*env)->CallVoidMethod(env,clazz,methodID);

}

9.c++ 开发JNI

===========

C的预处理命令

  • #开头的就是c/c++的预处理命令
    自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

最后

在此为大家准备了四节优质的Android高级进阶视频:

架构师项目实战——全球首批Android开发者对Android架构的见解

附相关架构及资料

image.png

往期Android高级架构资料、源码、笔记、视频。高级UI、性能优化、架构师课程、NDK、混合式开发(ReactNative+Weex)微信小程序、Flutter全方面的Android进阶实践技术,群内还有技术大牛一起讨论交流解决问题。

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

!**

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

最后

在此为大家准备了四节优质的Android高级进阶视频:

架构师项目实战——全球首批Android开发者对Android架构的见解

附相关架构及资料

[外链图片转存中…(img-RolrVIsU-1713227170466)]

往期Android高级架构资料、源码、笔记、视频。高级UI、性能优化、架构师课程、NDK、混合式开发(ReactNative+Weex)微信小程序、Flutter全方面的Android进阶实践技术,群内还有技术大牛一起讨论交流解决问题。

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

  • 3
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值