Android JNI编程(八)——体验AS2.2.2编写Jni程序、Java调C、C调Java函数、将C代码中的Log打印至Logcat
版权声明:本文出自阿钟的博客,转载请注明出处:http://blog.csdn.net/a_zhon/。
不得不说在AS2.2以上的版本进行开发就一个字——爽,在2.0上使用jni出现各种问题现在果断使用新版本开发了。
使用AS2.0创建Jni项目: 使用AndroidStudio编写第一个JNI程序
一:编写Jni程序
1. 我们创建一个项目,将Include C++ Support勾选一路Next直到Finish 这样就可以全自动为我们创建好一个带有Jni的项目了
编译过程中需要你为这个工程配置Ndk的路径,只需在
Project Sturcture配置即可(不懂看文章上面的链接文章)。如果你第一次使用,那么他会提示你需要下CMake你也只需下载即可
2. 在AS2.2以上版本中,使用的是CMake来编译我们的C代码
CMake官网:https://cmake.org/
Google官方文档:Jni开发文档
CMakeLists.txt
//以#号开头都是注释,这里把它们都删了也就更清楚了
cmake_minimum_required(VERSION 3.4.1)
#生成的库名
add_library( native-lib
#编译的文件路径
src/main/cpp/native-lib.c )
find_library( log-lib
log )
target_link_libraries( native-lib
${log-lib} )
3. 默认创建的是cpp文件,这里将它改成.c文件同时修改CMakeLists.txt中的src/main/cpp/native-lib.c文件后缀名,改完之后记得sync一下
4. 在build.gradle中配置生成的so库类型,记得sync一下
android {
//。。。
defaultConfig {
//。。。
ndk {
moduleName "native-lib"//生成so库的名字,与CMakeLists中的一致
abiFilters "armeabi", "armeabi-v7a", "x86"//cpu类型
}
}
}
二:Java调用C代码
1. 创建一个类,存放native函数
public class JniInterface {
//加载so文件
static {
System.loadLibrary("native-lib");
}
/*返回一个字符串*/
public native String stringFromJNI();
}
2. 使用Alt+Enter来自动生成对应的Jni函数,确定之后会在main/jni目录中创建一个C文件,我们只需要将里面的方法复制到我们的native-lib.c文件中
1. native-lib.c文件
#include <jni.h>
/**
* jstring: 返回值
* Java_全类名_方法名
* JNIEnv* env: 里面有很多方法
* jobject jobj: 谁调用了这个方法就是谁的实例
*/
jstring Java_com_zsy_ajni_JniInterface_stringFromJNI(JNIEnv *env, jobject jobj) {
char *str = "Hello from C";
return (*env)->NewStringUTF(env, str);
}
3. 在我们的MainActivity中调用stringFromJNI()便可以调用成功了。
然后在项目的build/intermediates/cmake/debug/obj/这里就可以看到编译生成的so库了
三:C调用Java代码
1. 在JniInterface类中继续增加我们的native方法,让java调用callbackAdd();函数,然后Jni调用add(int a,int b);函数
public class JniInterface {
//加载so文件
static {
System.loadLibrary("native-lib");
}
/**
* 返回一个字符串
*/
public native String stringFromJNI();
/**
* 当执行这个函数的时候,让C代码调用add(int a, int b)
*/
public native void callbackAdd();
/**
* 这个函数让C调用
*/
public int add(int a, int b) {
Log.d("TAG", "我是add函数被C调用了a = " + a + ";b = " + b);
return a + b;
}
}
2. MainActivity中调用callbackAdd()函数
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
JniInterface jni = new JniInterface();
TextView tv = (TextView) findViewById(R.id.sample_text);
tv.setText(jni.stringFromJNI());
//调用这个函数,让C调用我们的add函数
jni.callbackAdd();
}
}
3. 接下来就是重点了,在native-lib.c文件继续中生成callbackAdd()对应的Jni函数代码中来调用我们的add(int a, int b)函数
/**
* C代码调用Java JniInterface类中的add(int a, int b)函数
*/
void Java_com_zsy_ajni_JniInterface_callbackAdd(JNIEnv *env, jobject jobj) {
//1、得到字节码 FindClass(env,类的全路径);
jclass class = (*env)->FindClass(env, "com/zsy/ajni/JniInterface");
//2、得到方法 GetMethodID(env,jclass,方法名,方法签名);
//方法签名是要调用的方法名对应的签名
jmethodID methodID = (*env)->GetMethodID(env, class, "add", "(II)I");
//3、实例化该类
jobject obj = (*env)->AllocObject(env, class);
//4.调用Java类中的方法
//成功调用add方法;第二三个参数是我们调用add方法传入的参数。value就是add方法的返回值
jint value = (*env)->CallIntMethod(env, obj, methodID, 45, 23);
}
4. 上面代码中,我们需要去获取一个方法的签名那我们来获取一下:
将工程build让类生成对应的.class文件。然后使用
Terminal进入到app\build\intermediates\classes\debug>目录下,执行javap -s 类的全路径descriptor就是方法签名。
![]()
类的全路径可以通过在类名上右键,点击Copy Reference即可
5. 运行我们的程序,就可以在控制看到Log了也就证明方法调用成功
四:在C代码中打印Log至Logcat
上面我们是在Java函数中打印的Log,现在我们来让C中也可以打印Log
1.配置C代码打印Log
- 在build.gradle中添加日志打印so库
ndk {
//....
//就是依赖ndk工具包中的一个liblog.so文件
ldLibs "log"
}
- 在.c文件中添加如下代码,看注释就明白了
#include <android/log.h>
//日志标签,随意定义
#define LOG_TAG "TAG"
//Debug等级
#define LOGD(...)__android_log_print(ANDROID_LOG_DEBUG,LOG_TAG,__VA_ARGS__)
//Info等级
#define LOGI(...)__android_log_print(ANDROID_LOG_INFO,LOG_TAG,__VA_ARGS__)
//Error等级
#define LOGE(...)__android_log_print(ANDROID_LOG_ERROR,LOG_TAG,__VA_ARGS__)
- 配置好了,我们就可以来用了
#include <jni.h>
#include <android/log.h>
//日志标签,随意定义
#define LOG_TAG "TAG"
//Debug等级
#define LOGD(...)__android_log_print(ANDROID_LOG_DEBUG,LOG_TAG,__VA_ARGS__)
//Info等级
#define LOGI(...)__android_log_print(ANDROID_LOG_INFO,LOG_TAG,__VA_ARGS__)
//Error等级
#define LOGE(...)__android_log_print(ANDROID_LOG_ERROR,LOG_TAG,__VA_ARGS__)
/**
* jstring: 返回值
* Java_全类名_方法名
* JNIEnv* env: 里面有很多方法
* jobject jobj: 谁调用了这个方法就是谁的实例
*/
jstring Java_com_zsy_ajni_JniInterface_stringFromJNI(JNIEnv *env, jobject jobj) {
char *str = "Hello from C";
return (*env)->NewStringUTF(env, str);
}
/**
* C代码调用Java JniInterface类中的add(int a, int b)函数
*/
void Java_com_zsy_ajni_JniInterface_callbackAdd(JNIEnv *env, jobject jobj) {
//此处代码省略,与上面代码一致
jint value = (*env)->CallIntMethod(env, obj, methodID, 45, 23);
//调用Log输出,使用与printf函数一致
LOGD("value=%d", value);
}
2. 成功运行后机可以在Logcat查看到日志信息了
在2.2.2版本上可以关联到源码跟踪源码非常方便,写Jni也是个不错的体验。Demo地址
Android JNI编程(八)——体验AS2.2.2编写Jni程序、Java调C、C调Java函数、将C代码中的Log打印至Logcat的更多相关文章
- Android JNI编程(七)——使用AndroidStudio编写第一个JNI程序
版权声明:本文出自阿钟的博客,转载请注明出处:http://blog.csdn.net/a_zhon/. 目录(?)[+] 1.简单介绍一下NDK和JNI NDK:NDK是Native Develop ...
- Android中JNI编程的那些事儿(1)
转:Android中JNI编程的那些事儿(1)http://mobile.51cto.com/android-267538.htm Android系统不允许一个纯粹使用C/C++的程序出现,它要求必须 ...
- 【转】Android JNI编程—JNI基础
原文网址:http://www.jianshu.com/p/aba734d5b5cd 最近看到了很多关于热补的开源项目——Depoxed(阿里).AnFix(阿里).DynamicAPK(携程)等,它 ...
- 在 JNI 编程中避免内存泄漏
JAVA 中的内存泄漏 JAVA 编程中的内存泄漏,从泄漏的内存位置角度可以分为两种:JVM 中 Java Heap 的内存泄漏:JVM 内存中 native memory 的内存泄漏. Java H ...
- 【Android 应用开发】Android 网络编程 API笔记 - java.net 包 权限 地址 套接字 相关类 简介
Android 网络编程相关的包 : 9 包, 20 接口, 103 类, 6 枚举, 14异常; -- Java包 : java.net 包 (6接口, 34类, 2枚举, 12异常); -- An ...
- 【Android 应用开发】Android 网络编程 API笔记 - java.net 包相关 接口 api
Android 网络编程相关的包 : 9 包, 20 接口, 103 类, 6 枚举, 14异常; -- Java包 : java.net 包 (6接口, 34类, 2枚举, 12异常); -- An ...
- 在 JNI 编程中避免内存泄漏与崩溃
JNI 编程简介 JNI,Java Native Interface,是 native code 的编程接口.JNI 使 Java 代码程序可以与 native code 交互——在 Java 程序中 ...
- 解析Java的JNI编程中的对象引用与内存泄漏问题
JNI,Java Native Interface,是 native code 的编程接口.JNI 使 Java 代码程序可以与 native code 交互——在 Java 程序中调用 native ...
- Android 网络编程 API笔记 - java.net 包 权限 地址 套接字 相关类 简介
Android 网络编程相关的包 : 9 包, 20 接口, 103 类, 6 枚举, 14异常; -- Java包 : java.net 包 (6接口, 34类, 2枚举, 12异常); -- An ...
随机推荐
- 洛谷P3403跳楼机(最短路构造/同余最短路)
题目-> 解题思路: 最短路构造很神啊. 先用前两个值跑在第三个值模意义下的同余最短路(这步贪心可以证明,如果第三步长为z,那么如果n+z可以达到,n+2z同样可以达到) 最后计算与楼顶差多少个 ...
- 【Educational Codeforces Round 35 D】Inversion Counting
[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 排列中交换任意两个数字. 排列的逆序对个数的奇偶性会发生变化. 翻转这个过程其实就是len/2对数字发生交换. 交换了偶数次的话,不 ...
- 解决iOS空指针数据的问题
iOS开发中常常会遇到空指针的问题. 如从后台传回来的Json数据,程序中不做推断就直接赋值操作,非常有可能出现崩溃闪退. 为了解决空指针的问题,治标的方法就是遇到一个处理一个.这样业务代码里面就插了 ...
- textview-显示行数限制
在代码中直接添加 android:maxLines="2" android:ellipsize="end" 跟ellipsize搭配使用,超过两行的时候,第二行 ...
- php编译参数注释
1. 指定安装路径 --prefix=PREFIX 2. 指定运行用户 --with-fpm-user=nginx 3. 指定运行组 --with-fpm-group=nginx 3.与'--pref ...
- Javascript和jquery事件--键盘事件KeyboardEvent
Js和jq事件—键盘事件KeyboardEvent 键盘事件keydown,keypress和keyup,还需要涉及到一个文本事件textInput. keydown,keypress和keyup事件 ...
- Python 极简教程(十三)while 循环
循环简单来说就是让一段代码按你想要的方式多次运行.软件拥有强大的运算能力,就是由循环提供的. 在 Python 中支持的循环由两种:while 循环 和for 循环. 现在我们先来讲while循环. ...
- CSDN博客的文章分类和战略规划
CSDN原创文章已经有300多篇了,现在已经整理了好多个分类目录了. 今天,特别向大家介绍下,每个分类的含义和规划. CSDN博客是我的一个重要的自媒体,也是我的一个战略实践. 我会精心维护这个博客, ...
- 【CS Round #46 (Div. 1.5) A】Letters Deque
[链接]h在这里写链接 [题意] 在这里写题意 [题解] string类模拟 [错的次数] 0 [反思] 在这了写反思 [代码] /* */ #include <cstdio> #incl ...
- JS概述
从Asp.NET跨越到JavaScript.这既是一个新的领域也是一个非常熟悉的地方,新是由于不知道什么是JavaScript,首先来了解一下什么是JavaScript. ...