转 Android—JNI调用简单实例解析
2016-05-05 15:09
591 查看
转自:/article/4718657.html 感谢原作者的细心整理!
1. 在Eclipse中创建项目:TestJNI
2. 新创建一个class:TestJNI.java
?
以上代码声明三个本地方法。
3. 编译JNI
找到Android项目中bin目录下,会有classes文件夹,Eclipse自动为我们生成的字节码文件就在这个目录下。
我们在该路径下,使用javah命令,生成我们想要得到的.h头文件,如下图所示:
执行javah -jni com.wwj.jni.TestJNI命令之后,会在classes目录下生成头文件:com_wwj_jni_TestJNI.h
将它复制到jni文件夹下,打开如下:
?
以上代码就是通过javah命令生成jni层代码。
4. 使用C/C++实现JNI
在jni文件夹下,创建com_wwj_jni_TestJNI.h对应的cpp文件:com_wwj_jni_TestJNI.cpp
我们再添加两个文件Add.h,Add.cpp,具体实现放在这两个文件中来完成。
Add.h
?
Add.cpp
?
com_wwj_jni_TestJNI.cpp的实现:
5. 创建mk文件,并使用ndk-build命令生成.so动态链接库文件
在jni目录下创建Android.mk文件如下:
其中
LOCAL_PATH是C/C++代码所在目录,也就是我们的jni目录。
LOCAL_MODULE是要编译的库的名称。编译器会自动在前面加上lib,在后面加上.so。
LOCAL_SRC_FILES是要编译的C/C++文件。
然后我还需要在Android项目根目录下创建Application.mk文件:
写完了这两个mk文件,我们就可以用ndk来为我们生成相应的动态链接库了。前提你需要下载NDK,并把NDK路径配置到path环境变量中去,笔者配置的路径是:D:\Cocos2dx\android-ndk-r9d,具体视个人情况而定。
进入Application.mk文件所在目录,在命令行中使用ndk-build生成.so文件
编译成功后会在工程目录的libs/armeabi目录下生成一个libTestJNI.so文件。
项目结构会变成如下:
6. 在Java中调用JNI
运行项目,效果图如下:
那只剩下log 这一种方式, 那么多何在 C/C++ 里打印出Log呢 。
1. 导入log头文件
在你使用的 .c/ .cpp 文件中
导入 log.h 头文件
#include<Android/log.h>
2.在Android.mk 中
加上
LOCAL_LDLIBS :=-llog
注意Android.mk里有一行include $(CLEAR_VARS)
必须把LOCAL_LDLIBS :=-llog放在它后面才有用,
否则相当于没写。
3. 定义LOG 函数
先定义一个全局变量,再定义一些输出的LOG函数:
#define TAG "myDemo-jni" // 这个是自定义的LOG的标识
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG,TAG ,__VA_ARGS__) // 定义LOGD类型
上述代码中定义的函数
分别对应于Android 的Java代码中的
Log.d(), Log.i(), Log.w(),Log.e(), Log.f()等方法.
4.举例
#include <jni.h>
#include <string.h>
#include <android/log.h>
#define TAG "myhello-jni-test" // 这个是自定义的LOG的标识
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG,TAG,__VA_ARGS__) // 定义LOGD类型
JNIEXPORT jstring JNICALL Java_com_snail_helloworld_MainActivity_myhello(JNIEnv * env, jobject obj) {
int i = 0;
LOGD("########## i = %d", i);
return env->NewStringUTF("Hello From CPP");
}
转自:/article/4718657.html 感谢原作者的细心整理!
1. 在Eclipse中创建项目:TestJNI
2. 新创建一个class:TestJNI.java
?
3. 编译JNI
找到Android项目中bin目录下,会有classes文件夹,Eclipse自动为我们生成的字节码文件就在这个目录下。
我们在该路径下,使用javah命令,生成我们想要得到的.h头文件,如下图所示:
执行javah -jni com.wwj.jni.TestJNI命令之后,会在classes目录下生成头文件:com_wwj_jni_TestJNI.h
将它复制到jni文件夹下,打开如下:
?
4. 使用C/C++实现JNI
在jni文件夹下,创建com_wwj_jni_TestJNI.h对应的cpp文件:com_wwj_jni_TestJNI.cpp
我们再添加两个文件Add.h,Add.cpp,具体实现放在这两个文件中来完成。
Add.h
?
?
#include <stdio.h> #include <stdlib.h> #include "com_wwj_jni_TestJNI.h" #include "Add.h" Add *pCAdd = NULL; JNIEXPORT jboolean JNICALL Java_com_wwj_jni_TestJNI_Init(JNIEnv *env,jobject obj) { if (pCAdd == NULL) { pCAdd = new CAdd; } return pCAdd != NULL; } JNIEXPORT jint JNICALL Java_com_wwj_jni_TestJNI_Add(JNIEnv *env, jobject obj, jint x, jint y) { int res = -1; if (pCAdd != NULL) { res = pCAdd->Add(x, y); } return res; } JNIEXPORT void JNICALL Java_com_wwj_jni_TestJNI_Destory(JNIEnv *env, jobject obj) { if (pCAdd != NULL) { pCAdd = NULL; } }
5. 创建mk文件,并使用ndk-build命令生成.so动态链接库文件
在jni目录下创建Android.mk文件如下:
LOCAL_PATH := $(call my-dir) include $(CLEAR_VARS) LOCAL_MODULE := TestJNI LOCAL_SRC_FILES := com_wwj_jni_TestJNI.cpp LOCAL_SRC_FILES += Add.cpp include $(BUILD_SHARED_LIBRARY)
其中
LOCAL_PATH是C/C++代码所在目录,也就是我们的jni目录。
LOCAL_MODULE是要编译的库的名称。编译器会自动在前面加上lib,在后面加上.so。
LOCAL_SRC_FILES是要编译的C/C++文件。
然后我还需要在Android项目根目录下创建Application.mk文件:
APP_PROJECT_PATH := $(call my-dir) APP_MODULES := TestJNI
写完了这两个mk文件,我们就可以用ndk来为我们生成相应的动态链接库了。前提你需要下载NDK,并把NDK路径配置到path环境变量中去,笔者配置的路径是:D:\Cocos2dx\android-ndk-r9d,具体视个人情况而定。
进入Application.mk文件所在目录,在命令行中使用ndk-build生成.so文件
编译成功后会在工程目录的libs/armeabi目录下生成一个libTestJNI.so文件。
项目结构会变成如下:
6. 在Java中调用JNI
package com.wwj.jni; import android.os.Bundle; import android.widget.TextView; import android.app.Activity; public class TestJNIActivity extends Activity { private TextView textView; static { // 加载动态库 System.loadLibrary("TestJNI"); } @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); textView = (TextView) findViewById(R.id.textview); TestJNI testJNI = new TestJNI(); // 调用native方法 boolean init = testJNI.Init(); if (init == true) { // 调用Add函数 int sum = testJNI.Add(100, 150); textView.setText("你真是个" + sum); } else { textView.setText("你比二百五还要二百五"); } testJNI.Destory(); } }
运行项目,效果图如下:
Jni中 log 的打印:
Jni 本地方法是编译出 .so 库给JAVA 程序调用的,所有无法断点调试。那只剩下log 这一种方式, 那么多何在 C/C++ 里打印出Log呢 。
1. 导入log头文件
在你使用的 .c/ .cpp 文件中
导入 log.h 头文件
#include<Android/log.h>
2.在Android.mk 中
加上
LOCAL_LDLIBS :=-llog
注意Android.mk里有一行include $(CLEAR_VARS)
必须把LOCAL_LDLIBS :=-llog放在它后面才有用,
否则相当于没写。
3. 定义LOG 函数
先定义一个全局变量,再定义一些输出的LOG函数:
#define TAG "myDemo-jni" // 这个是自定义的LOG的标识
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG,TAG ,__VA_ARGS__) // 定义LOGD类型
上述代码中定义的函数
分别对应于Android 的Java代码中的
Log.d(), Log.i(), Log.w(),Log.e(), Log.f()等方法.
4.举例
#include <jni.h>
#include <string.h>
#include <android/log.h>
#define TAG "myhello-jni-test" // 这个是自定义的LOG的标识
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG,TAG,__VA_ARGS__) // 定义LOGD类型
JNIEXPORT jstring JNICALL Java_com_snail_helloworld_MainActivity_myhello(JNIEnv * env, jobject obj) {
int i = 0;
LOGD("########## i = %d", i);
return env->NewStringUTF("Hello From CPP");
}
相关文章推荐
- Android相册中获取图片和路径
- android自定义圆角Dialog
- Android中什么是anr ? 原因 or 解决方法
- Android Studio单元测试入门
- Android-Universal-Image-Loader三大组件DisplayImageOptions、ImageLoader、ImageLoaderConfiguration详解
- android:layout_gravity和android:gravity属性的区别
- android在listview中放入从sdcard读取的bitmap
- Android计时器
- Android Q&A | Incorrect line ending: found carriage return (\r) without corresponding newline (\n)
- android从sdcard中读取bitmap
- android studio使用小技巧
- Android Animation动画(很详细)
- android保存bitmap到sdcard
- android aidl文件无法自动生成java文件
- android studio导入第三方库(此处是slidingmenu)
- 百度地图
- Android发布app到google play时被拒绝解决方法
- Android 平板上 同时显示多个进度框 正在加载中
- Android数据持久化存储
- Android Studio 自定义View命名空间报错的解决方法