android开发——关于编写JNI

来源:互联网 发布:echart折线图动态数据 编辑:程序博客网 时间:2024/06/13 02:16
最近由于要开发android支付应用,实现刷卡读取磁卡的数据功能,需要编写JNI调用C语言底层库,在学习过程中也遇到了一些困难和问题,在这里记录下来,希望能给遇到同样问题的朋友提供帮助,避免走弯路。通过一个简单的调用c语言输出“hello”语句的例子来介绍如何编写JNI。
工程如下:


TestActivity.java:调用JNI方法,输出hello语句。
JniTest.java: 编写native方法,调用C语言方法,让TestActivity.java调用。
jni:在创建工程的时候自行创建,放编译好的so动态链接库。

1.在android工程中写native方法。
文件JniTest.java
package com.android.jni;public class JniTest {     public static native String hello ();}

2. 编译h头文件(windows环境下)
打开控制台,进入工程目录(F:\androidDemo\test)
cd F:\android\test

输入如下命令编译h头文件
javah -classpath bin /classes -d jni com.android.jni.JniTest


-classpath ——类路径 bin/classes
-d ——保存目录:jni
com.android.jni.JniTest:包名+类名

这时候jni文件夹下就多出了一个h头文件——com_android_jni_JniTest.h。

3.编写C文件。
新建一个C文件——JniTest.c,实现com_android_jni_JniTest.h里的方法。
文件JniTest.c:
#include "com_android_jni_JniTest.h"#include <stdio.h>/* * Class:     com_android_jni_JniTest * Method:    hello * Signature: ()Ljava/lang/String; */JNIEXPORT jstring JNICALL Java_com_android_jni_JniTest_hello  (JNIEnv * env, jclass cla){return (*env)->NewStringUTF(env, "hello");  }

注:在h头文件中没有写上参数名,如env和cla,在c文件需要补上。

4.编写Android.mk文件。
在jni目录下新建Android.mk文件,内容如下:
LOCAL_PATH := $(call my-dir)include $(CLEAR_VARS)LOCAL_MODULE    := JniTestLOCAL_SRC_FILES := JniTest.cinclude $(BUILD_SHARED_LIBRARY)

该文件中的一些变量对应的含义如下:
LOCAL_SRC_FILES -编译的源文件
LOCAL_MODULE -编译的目标对象

4.编译so动态链接库
由于编译so动态链接库需要Linux环境,如果你的操作系统是windows,可以安装cygwin模拟Linux环境,然后安装NDK即可,如果你是Linux环境,那么恭喜你,可以省略一步,直接安装NDK即可,若你是ubuntu环境,那么可以直接参考我之前的文章,(Ubuntu环境下配置NDK)其他环境就需要你自己google下了,过程应该大同小异了。
进入test工程(由于NDK配置路径问题,我将工程拷到ndk目录下的samples里)(F:\android\android-ndk-r7b\samples\test)

输入编译so命令
$NDK/ndk-build

若出现如上显示,则代表编译成功。

5.加载so文件
在JniTest.java中添加加载so文件代码,具体代码如下:
package com.android.jni;public class JniTest {     static {            System. loadLibrary("JniTest"); //加载so动态链接库           }      public static native String hello();}

在JniTest.java调用hello方法,具体代码如下:
package com.android.test;import com.android.jni.JniTest;import android.app.Activity;import android.os.Bundle;import android.widget.TextView;public class TestActivity extends Activity {     private TextView tv;    /** Called when the activity is first created. */    @Override    public void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout. main);        tv=(TextView)findViewById(R.id. tv);        tv.setText(JniTest.hello());    }}

main.xml:
<?xml version="1.0" encoding= "utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"    android:layout_width= "fill_parent"    android:layout_height= "fill_parent"    android:orientation="vertical" >    <TextView        android:id= "@+id/tv"        android:layout_width= "fill_parent"        android:layout_height= "wrap_content"        android:text= "@string/hello" /></LinearLayout>

运行结果如下:


大功告成,若有其他不明白的地方,随时和我联系,我一定尽力帮助,大家互相学习!若有不对的地方,也请各位前辈们指正,谢谢!

在编写JNI的过程中,也遇到了一些问题,编译不成功,问题和解决办法如下:
问题1.
Android NDK:Your APP_BUILD_SCRIPT points to an unknow files: ./jni/Android.mk
若出现该问题,是由于没有编写Android.mk文件。

问题2.
arm-linux-androideabi-gcc.exe:CreateProcess: no such file or directory
可能是内存溢出问题,只要关闭eclipse或者占内存很大的软件即可。

问题3.
error:parameter name omitted

方法缺少参数名。由于h头文件是没有参数名的,所以很容易在C文件忘记加上,例如:
JNIEXPORT jstring JNICALL Java_com_android_jni_JniTest_hello  (JNIEnv *, jclass){     return (*env)->NewStringUTF(env, "hello");  }

原创粉丝点击