Activity
public class MainActivity extends ListActivity {static {System.loadLibrary("hello");// 在java代码中引入libs目录下的库函数,文件名为【libhello.so】。注意,引入时的文件名要去掉前面的lib和后面的.soSystem.loadLibrary("hellocpp");}
private TextView tv_info;protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);String[] array = { "调用C中的无参方法,返回一个字符串", //"调用C中的有参方法,返回处理1+2后的值",//"对字符串进行处理", //"对int数组进行处理",//"调用C++中的无参方法,返回一个字符串", };tv_info = new TextView(this);tv_info.setTextColor(Color.BLUE);tv_info.setTextSize(TypedValue.COMPLEX_UNIT_SP, 16);tv_info.setPadding(20, 10, 20, 10);getListView().addFooterView(tv_info);setListAdapter(new ArrayAdapter<String>(this, android.R.layout.simple_list_item_1, new ArrayList<String>(Arrays.asList(array))));}
@Overrideprotected void onListItemClick(ListView l, View v, int position, long id) {switch (position) {case 0:String stringFromC = MyNativeMethods.getInstance().helloFromC();tv_info.setText(stringFromC);break;case 1:int intFromC = MyNativeMethods.getInstance().passwordFromC(1, 2);tv_info.setText(intFromC + "");break;case 2:tv_info.setText(MyNativeMethods.getInstance().encodeFromC("abc123", 1));break;case 3:int array[] = MyNativeMethods.getInstance().intMethod(new int[] { 1, 2, 3, });tv_info.setText(Arrays.toString(array));break;case 4:tv_info.setText(MyNativeMethods.getInstance().helloFromCPP());break;}}}
本地方法
/** 存放native方法的类 */public class MyNativeMethods {/**返回一个字符串*/public native String helloFromC();/**返回int类型计算后的结果*/public native int passwordFromC(int x, int y);/**对给定字符串进行处理,模拟加密运算*/public native String encodeFromC(String text, int length);/**给c代码传递int数组,让c代码给这个数组进行操作;模拟多媒体解码*/public native int[] intMethod(int[] iNum);/**c++中的方法*/public native String helloFromCPP();//******************************************************************************************private static MyNativeMethods mEmployee;private MyNativeMethods() {}public static MyNativeMethods getInstance() {if (mEmployee == null) {mEmployee = new MyNativeMethods();}return mEmployee;}}
c文件
#include <stdio.h>#include <stdlib.h>#include <jni.h>//必须添加的头文件#include <string.h>#include "com_bqt_hellofromc_MyNativeMethods.h"//引入生成的头文件,注意要把此文件放在jni文件夹中,src目录下的此文件不需要了//在C中使用log,除在此声明外,还需在Android.mk文件中增加【LOCAL_LDLIBS += -llog】#include <android/log.h>//引入log头文件#define LOG_TAG "bqt"// 这个是自定义的LOG的标识//定义一些输出的LOG函数#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG,LOG_TAG ,__VA_ARGS__) // 定义LOGD类型#define LOGI(...) __android_log_print(ANDROID_LOG_INFO,LOG_TAG ,__VA_ARGS__) // 定义LOGI类型#define LOGW(...) __android_log_print(ANDROID_LOG_WARN,LOG_TAG ,__VA_ARGS__) // 定义LOGW类型#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR,LOG_TAG ,__VA_ARGS__) // 定义LOGE类型#define LOGF(...) __android_log_print(ANDROID_LOG_FATAL,LOG_TAG ,__VA_ARGS__) // 定义LOGF类型//******************************************************************************************【1】返回一个字符串JNIEXPORT jstring JNICALL Java_com_bqt_hellofromc_MyNativeMethods_helloFromC(JNIEnv *env, jobject obj) {//写成【JNIEnv * env】或【JNIEnv *env】或【JNIEnv*env】都可以。【返回值】【方法名】【参数列表】返回值类型jstring就是java中的stringchar* cstr = "hello from c"; // char* 在c中可用来表示一个字符串。注意,这里绝对不能有中文jstring jstr = (*env)->NewStringUTF(env, cstr);return jstr;}//******************************************************************************************【2】返回int类型计算后的结果JNIEXPORT jint JNICALL Java_com_bqt_hellofromc_MyNativeMethods_passwordFromC(JNIEnv * env, jobject obj, jint a, jint b) {return a + b + 10000; //c中的int占用字节数在不同环境下可能不同,可能是0-65535,所以,稍微大一点的数(十万级别)都得用double}//******************************************************************************************【3】对给定字符串进行处理,模拟加密运算JNIEXPORT jstring JNICALL Java_com_bqt_hellofromc_MyNativeMethods_encodeFromC(JNIEnv * env, jobject obj, jstring passwd, jint length) {//char* cstr = Jstring2CStr2(env, passwd); //将java的字符串转化为c语言。卧槽这方法编译不通过,估计是因为找不到一些库之类的东西的原因!// int i = 0;// for (i = 0; i < length; i++) {// *(cstr + i) += 1; //给C语言字符加1// }char* cstr = "abc";return (*env)->NewStringUTF(env, cstr); //将c语言字符串转化为java字符串}//******************************************************************************************【4】对给定数组进行操作,模拟多媒体解码JNIEXPORT jintArray JNICALL Java_com_bqt_hellofromc_MyNativeMethods_intMethod(JNIEnv * env, jobject jobject, jintArray arr) {int len = (*env)->GetArrayLength(env, arr); // 获取数组的长度LOGI("ArrayLength=%d", len);//LOG中也不能有中文jint* p = (*env)->GetIntArrayElements(env, arr, 0); //取出数组中第一个元素的内存地址//If isCopy is not NULL, then *isCopy is set to JNI_TRUE if a copy ismade; if no copy is made, it is set to JNI_FALSE.int i = 0;for (; i < len; i++) {LOGI("before-%d-=%d", i, *(p+i));//这一步没问题,会一个个打印所有数组中的内容*(p + i) += 5; //取出的每个元素加5LOGI("after-%d-=%d", i, *(p+i));//这一步也没问题,会一个个打印处理后的内容}jint* elems = (*env)->GetIntArrayElements(env, arr, 0); //再次获取元素的内存地址LOGI("ok-%d-=%d", i, *elems);//但返回的为啥还是未经处理的?//重新创建一个数组jintArray array = (*env)->NewIntArray(env, len); //Construct构造 a new primitive array object基本数组对象int ii = 0;int jj;for (; ii < len; ii++) {jj = 10000 + ii;(*env)->SetIntArrayRegion(env, array, ii, 1, &jj); //Sets the value of one element in a primitive array.其中【&】代表【取地址】操作,而非逻辑操作//(*env)->SetIntArrayRegion(env,array,【start】,【len】,buffer) , 从start开始复制长度为len 的数据 buffer到 array 中}return array;}//******************************************************************************************【工具方法】把java的字符串转换成c的字符串char* Jstring2CStr(JNIEnv* env, jstring jstr) {char* rtn = NULL;//1:先找到字节码文件jclass clsstring = (*env)->FindClass(env, "Java/lang/String"); //Java或java都一样jstring strencode = (*env)->NewStringUTF(env, "GB2312"); //和UTF-8不行//2:通过字节码文件找到方法IDjmethodID mid = (*env)->GetMethodID(env, clsstring, "getBytes", "(Ljava/lang/String;)[B");//3:通过方法id,调用方法jbyteArray barr = (jbyteArray)(*env)->CallObjectMethod(env, jstr, mid, strencode); // String .getByte("GB2312");//4:得到数据的长度jsize alen = (*env)->GetArrayLength(env, barr);//5:得到数据的首地址jbyte* ba = (*env)->GetByteArrayElements(env, barr, JNI_FALSE);//6:得到C语言的字符串if (alen > 0) {rtn = (char*) malloc(alen + 1); //"\0"memcpy(rtn, ba, alen);rtn[alen] = 0;}(*env)->ReleaseByteArrayElements(env, barr, ba, 0); //return rtn;}
c++文件
#include <stdio.h>#include <stdlib.h>#include <jni.h>#include "com_bqt_hellofromc_MyNativeMethods.h"//引入生成的头文件,这个东西在这里又必须要了!JNIEXPORT jstring JNICALL Java_com_bqt_hellofromc_MyNativeMethods_helloFromCPP(JNIEnv *env, jobject obj) {return env->NewStringUTF("hello c++");}
Android.mk文件
LOCAL_PATH := $(call my-dir)# C/C++代码所在目录,也就是我们的jni目录,不必修改include $(CLEAR_VARS)LOCAL_LDLIBS += -llog#若要在C中使用log,①需在.c中添加一系列声明,②还需在Android.mk文件中增加【LOCAL_LDLIBS += -llog】#必须把LOCAL_LDLIBS :=-llog放在【include $(CLEAR_VARS)】后面才有用LOCAL_MODULE := hello# 对应打包成函数库的名字,编译器会自动在前面加上lib,在后面加上.soLOCAL_SRC_FILES := hello.c# 对应c代码的文件 hello.c#不知道能不能同时编译多个#LOCAL_MODULE := hellocpp#LOCAL_SRC_FILES := hello.cppinclude $(BUILD_SHARED_LIBRARY)
Application.mk文件
APP_ABI := armeabi armeabi-v7a x86#Application.mk文件的目的是,描述在你的应用程序中所需要的模块(即静态库或动态库)#APP_ABI 的值以空格区分,代表要支持的架构,默认为【armeabi】。其他架构,ARMv7 【armeabi-v7a】;IA-32 【 x86】#每增加一个架构,编译后都会在lib目录下生成一个相应的文件夹,文件夹下的文件都是同名的.so文件(当然文件内容不一样)
javah生成的.h头文件
/* DO NOT EDIT THIS FILE - it is machine generated */#include <jni.h>/* Header for class com_bqt_hellofromc_MyNativeMethods */#ifndef _Included_com_bqt_hellofromc_MyNativeMethods#define _Included_com_bqt_hellofromc_MyNativeMethods#ifdef __cplusplusextern "C" {#endif/** Class: com_bqt_hellofromc_MyNativeMethods* Method: helloFromC* Signature: ()Ljava/lang/String;*/JNIEXPORT jstring JNICALL Java_com_bqt_hellofromc_MyNativeMethods_helloFromC(JNIEnv *, jobject);/** Class: com_bqt_hellofromc_MyNativeMethods* Method: passwordFromC* Signature: (II)I*/JNIEXPORT jint JNICALL Java_com_bqt_hellofromc_MyNativeMethods_passwordFromC(JNIEnv *, jobject, jint, jint);/** Class: com_bqt_hellofromc_MyNativeMethods* Method: encodeFromC* Signature: (Ljava/lang/String;I)Ljava/lang/String;*/JNIEXPORT jstring JNICALL Java_com_bqt_hellofromc_MyNativeMethods_encodeFromC(JNIEnv *, jobject, jstring, jint);/** Class: com_bqt_hellofromc_MyNativeMethods* Method: intMethod* Signature: ([I)[I*/JNIEXPORT jintArray JNICALL Java_com_bqt_hellofromc_MyNativeMethods_intMethod(JNIEnv *, jobject, jintArray);/** Class: com_bqt_hellofromc_MyNativeMethods* Method: helloFromCPP* Signature: ()Ljava/lang/String;*/JNIEXPORT jstring JNICALL Java_com_bqt_hellofromc_MyNativeMethods_helloFromCPP(JNIEnv *, jobject);#ifdef __cplusplus}#endif#endif