JNI NDK (AndroidStudio+CMake )实现C C++调用Java代码流程
JNI/NDK Java调用C/C++前言
通过第三篇文章讲解在实际的开发过程中Java层调用C/C++层的处理流程。其实我们在很大的业务里也需要C/C+ +层去调用Java层,这两层之间的相互调用显得如此的重要,正式两层之间的相互调用使得程序更具有高效性、安全性可言。下面主要讲解一下C/C+ +层调用Java层的处理流程。
JNI/NDK Java调用C/C++ 编写java文件
同样我们也需要先写java文件,用来让C/C++调用java层的方法,实现具体的业务逻辑。
public class NativeUtils {
//1、引用JIN/NDk库文件(库名称与创建的C/C++文件名保持一致)
static {
System.loadLibrary("jni-utils");
}
//2、定义native 原生方法 (代表该方法会调用C/C++来实现功能)
//有返回值、无参数 处理字符串
public native String JavaCallJNI();
//有参数、有返回值 处理int类型
public native int JavaCallJNISum(int num1, int num2);
//有参数、有返回值 处理int[] 数组类型
public native int[] JavaCallJNIArr(int[] arr);
//C/C++层调用该方法的回调
public native void JNICallJavaBack();
//java层方法的具体实现
public void JNICallJava(String msg) {
Log.e("TAG", "JNICallJava--->" + msg);
}
//C/C++层调用该方法的回调
public native void JNICallJavaSumBack();
//java层方法的具体实现
public void JNICallJavaSum(int num1, int num2) {
Log.e("TAG", String.format("JNICallJavaSum--->%d+%d=%d", num1, num2, num1 + num2));
}
//C/C++层调用该方法的回调
public native void JNICallJavaStaticMethodBack();
//java层方法的具体实现
public static void JNICallJavaStaticMethod() {
Log.e("TAG", "JNICallJavaStaticMethod--->");
}
}
JNI/NDK Java调用C/C++ 编写C/C++文件
通过上一步我们编写了java层的代码,其次我们需要在C/C++文件中进行调用java层的方法。这是一个比较抽象的处理。总体思想就是采用反射机制拿到方法的信息。


//1.引入Jni头文件
#include <jni.h>
#include <stdlib.h>
#include <stdio.h>
#include <android/log.h> //处理日志打印
//------------------------------------日志处理-----------------------------------
#define LOG_TAG "JNILogTag"
//不带格式log
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO,LOG_TAG,"%s",__VA_ARGS__)
//带格式
#define LOG_I(format, ...) __android_log_print(ANDROID_LOG_INFO,LOG_TAG,format,__VA_ARGS__) //2.编写NativeUtils对应的JNI的C/C++函数
//------------------------------C/C++函数解释--------------------------------------
//JNIEXPORT JNI导出 jstring 函数返回值 JNICALL JNI进行调用
//Java_全类名_NativeUtils方法名(JNIEnv *env,jobject jobject)
// JNIEnv *env C/C++中的函数指针 jobject jobject 调用Native方法的类对象 //------------------------------------sig签名处理-----------------------------------
//方式一:命令
//生成方法签名的方式:进行生成.class文件的目录下 执行: javap -s xxx.class
//方式二:规律 public String JNICallJava(String msg) java层的方法
//首先是参数(String msg) ---> (Ljava/lang/String;)
//其次返回值 String ----> Ljava/lang/String;
//最终签名 (Ljava/lang/String;)Ljava/lang/String;
//------------------------常见的转换表------------------------
// String Ljava/lang/String;
// int I
// int[] [I
// void V
//------------------------------------C/C++调用Java-----------------------------------
extern "C"
JNIEXPORT void JNICALL
Java_com_aynu_androidjni_NativeUtils_JNICallJavaBack(JNIEnv *env, jobject instance) {
//1.得到类的字节码 (调用java方法所在的类 包名+类名)
jclass cls = env->FindClass("com/aynu/androidjni/NativeUtils");
//2.获取方法id
//clazz 类的字节码 name java方法名称 sig java方法签名
jmethodID mid = env->GetMethodID(cls, "JNICallJava", "(Ljava/lang/String;)V");
//3.实例化该类
jobject jobject = env->AllocObject(cls);
//5.设置java层参数的值
jstring str = env->NewStringUTF("C/C++ input value");
//4.调用java层方法
env->CallVoidMethod(jobject, mid, str);
}
extern "C" JNIEXPORT void JNICALL
Java_com_aynu_androidjni_NativeUtils_JNICallJavaSumBack(JNIEnv *env, jobject instance) {
//1.得到类的字节码 (调用java方法所在的类 包名+类名)
jclass cls = env->FindClass("com/aynu/androidjni/NativeUtils");
//2.获取方法id
//参数解析 jmethodID GetMethodID(jclass clazz, const char* name, const char* sig)
//clazz 类的字节码 name java方法名称 sig java方法签名
jmethodID mid = env->GetMethodID(cls, "JNICallJavaSum", "(II)V");
//3.实例化该类
jobject jobject = env->AllocObject(cls);
//5.设置java层参数的值
jint num1 = ;
jint num2 = ;
//4.调用java层方法
env->CallVoidMethod(jobject, mid, num1, num2);
}
extern "C" JNIEXPORT void JNICALL
Java_com_aynu_androidjni_NativeUtils_JNICallJavaStaticMethodBack(JNIEnv *env, jobject instance) {
//1.得到类的字节码 (调用java方法所在的类 包名+类名)
jclass cls = env->FindClass("com/aynu/androidjni/NativeUtils");
//2.获取方法id
//参数解析 jmethodID GetMethodID(jclass clazz, const char* name, const char* sig)
//clazz 类的字节码 name java方法名称 sig java方法签名
jmethodID mid = env->GetStaticMethodID(cls, "JNICallJavaStaticMethod", "()V");
//3.实例化该类 该方法为static静态方法 故不需要实例化
//jobject jobject = env->AllocObject(cls);
//4.调用java层方法
env->CallStaticVoidMethod(cls, mid);
} //------------------------------------Java调用C/C++-----------------------------------
extern "C"
JNIEXPORT jintArray JNICALL
Java_com_aynu_androidjni_NativeUtils_JavaCallJNIArr(JNIEnv *env, jobject jobject,
jintArray arr_) {
//1.获取arr数组的元素
jint *arr = env->GetIntArrayElements(arr_, NULL);
//2.获取arr数组的长度
jsize arrSize = env->GetArrayLength(arr_);
//3.遍历数组
for (int i = ; i < arrSize; ++i) {
*(arr + i) += ;
}
//4.释放内存
env->ReleaseIntArrayElements(arr_, arr, );
//5.返回数组
return arr_;
} extern "C"
JNIEXPORT jint JNICALL
Java_com_aynu_androidjni_NativeUtils_JavaCallJNISum(JNIEnv *env, jobject jobject,
jint num1, jint num2) {
//1.相应的逻辑运算 return num1 + num2; } extern "C"
JNIEXPORT jstring JNICALL
Java_com_aynu_androidjni_NativeUtils_JavaCallJNI(JNIEnv *env, jobject jobject) { //3.编写具体的业务逻辑 return env->NewStringUTF("C/C++ Say");
}
JNI/NDK 进行调用
我们编写好java层和C/C++层之后,就需要我们进行去调用。

public class MainActivity extends AppCompatActivity {
private TextView mMsgTxt;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
initView();
}
private void initView() {
mMsgTxt = (TextView) findViewById(R.id.msg_txt);
NativeUtils nativeUtils = new NativeUtils();
//Java调用JNI
String msg = nativeUtils.JavaCallJNI();
//mMsgTxt.setText(msg);
//Java调用JNI实现两个数之和
int sum = nativeUtils.JavaCallJNISum(, );
//mMsgTxt.setText(String.format("10+5=%d", sum));
//Java调用JNI实现数组中的每个元素加10
int[] arr = new int[]{, , , , };
int[] jniArr = nativeUtils.JavaCallJNIArr(arr);
StringBuilder buffer = new StringBuilder();
for (int aJniArr : jniArr) {
buffer.append(aJniArr).append(",");
}
//mMsgTxt.setText(buffer.toString());
//JNICallJavaBack执行
nativeUtils.JNICallJavaBack();
//JNICallJavaSumBack执行
nativeUtils.JNICallJavaSumBack();
//JNICallJavaStaticMethodBack执行
nativeUtils.JNICallJavaStaticMethodBack();
}
}
JNI/NDK 结果
调用之后我们会生成对应的so库文件,同时也会展示我们的最后结果。

JNI/NDK 结束语
以上便是采用Androidstudio+CMake进行搭建JNI/NDK开发中C/ C++调用Java代码流程的项目。如若有理解错误的地方,请多多留言指教。
JNI NDK (AndroidStudio+CMake )实现C C++调用Java代码流程的更多相关文章
- C#调用Java代码
c#直接调用java代码,需要ikvmbin-0.44.0.5.zip.下载地址: http://pan.baidu.com/share/link?shareid=3996679697&uk= ...
- C#调用java代码(IKVMC)
参考资料:https://blog.csdn.net/threadroc/article/details/51406587 参考1:http://www.cnblogs.com/Jack-Blog/p ...
- python 调用 java代码
一.JPype简述 1.JPype是什么? JPype是一个能够让 python 代码方便地调用 Java 代码的工具,从而克服了 python 在某些领域(如服务器端编程)中的不足. 2.JPype ...
- python2.7 (x64) 调用 java 代码实践
背景:公司实施接口自动化测试,使用的框架python+unittest:因为开发使用的架构师SpringBoot,在测试过程中难免需要使用到python调用JAVA的一些接口,所以需要用到python ...
- Python调用Java代码部署及初步使用
Python调用Java代码部署: jpype下载地址:https://www.lfd.uci.edu/~gohlke/pythonlibs/#jpype 下载的时候需要使用Chrome浏览器进行下载 ...
- python调用Java代码,完毕JBPM工作流application
1.缘由 有一庞大Python django webproject,要引入工作流引擎,像OA一样.方便的流程控制与管理.Python或django关于工作流的开源插件,稀少,并且弱爆了,终于选用jav ...
- C代码调用Java代码
C代码调用Java代码应用场景 复用已经存在的java代码 c语言需要给java一些通知 c代码不方便实现的逻辑(界面) 反射 //1.加载类字节码 Class clazz = Demo.class. ...
- JNI系列——C文件中的方法调用Java中方法
1.创建xxx.jni包并在该包下实现一些Java的方法,和要调用的本地方法 2.实现MainActivity中的按钮点击事件-即点击按钮调用本地的方法 3.在C文件中的方法中回调Java的方法 3. ...
- 基于 Android NDK 的学习之旅----- C调用Java
许多成熟的C引擎要移植到Android 平台上使用 , 一般都会 提供 一些接口, 让Android sdk 和 jdk 实现. 下文将会介绍 C 如何 通过 JNI 层调用 Java 的静态和非静态 ...
随机推荐
- python访问互联网
1.python有一个网络包urllib,里面有很多网络模块,其中我们常用的就是urllib.request (module)这个模块 2.引入要是用的模块:import urllib.request ...
- Python You-Get (送你一个免广告的视频和音乐网站 VIP)
You-get可以在仅仅提供URL情况下就可以实现下载视频.图片.音乐等信息.也可以通过播放器在线观看视频或听音乐,重要的是再也不用烦恼弹出的广告了,如果你想观看视频,但又不想观看广告,并且你还想把视 ...
- C语言小笔记
头文件的书写 头文件实现函数声明,在使用模板后可以实现一个C文件中即使重复包含某个头文件,在系统中用于只会确认为一个包含 头文件包含可以理解为将头文件内容替换#include“...”行 模板(don ...
- [原创]K8Cscan插件之Weblogic漏洞扫描&通用GetShell Exploit
[原创]K8 Cscan 大型内网渗透自定义扫描器 https://www.cnblogs.com/k8gege/p/10519321.html Cscan简介:何为自定义扫描器?其实也是插件化,但C ...
- Retrofit2 完全解析 探索与okhttp之间的关系
转载请标明出处: http://blog.csdn.net/lmj623565791/article/details/51304204: 本文出自:[张鸿洋的博客] 之前写了个okhttputils的 ...
- 从session原理出发解决微信小程序的登陆问题
声明:本文为作者原创文章,转载请注明出处 https://www.cnblogs.com/MaMaNongNong/p/9127416.html 原理知识准备 对于已经熟悉了session原理的同 ...
- 前端JS 与 后台C# 之间JSON序列化与反序列化(笔记)
JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式. 在 前端浏览器 和 后端服务器 之间通常会使用 JSON格式的数据 来进行数据交互,而JSON格式的 ...
- 基于vue与vux做的可滑动tab组件(附源码)
背景 前不久,刚完成了一个商品列表+购物车功能的页面,因为一级商品分类在顶部tab中显示,可滑动,间距可定制,如下图所示: 定制的tab需求如下: 1. 每个tab-item的间距是相同的,可定制 2 ...
- μC/OS-II 任务就绪表及任务调度
任务调度 多任务操作系统的核心工作就是任务调度. 所谓调度,就是通过一个算法在多个任务中确定该运行的任务,做这项工作的函数就叫做调度器. μC/OS-II 进行任务调度的思想是 "近似地每时 ...
- MySQL之实现Oracle中的rank()函数的功能
假设表格为student, 数据如下: 我们要在MySQL中实现Oracle中的rank()函数功能,即组内排序,具体来说: 就是对student表中按照课程(course)对学生(name) ...