最近由于要开发android支付应用,实现刷卡读取磁卡的数据功能,需要编写JNI调用C语言底层库,在学习过程中也遇到了一些困难和问题,在这里记录下来,希望能给遇到同样问题的朋友提供帮助,避免走弯路。通过一个简单的调用c语言输出“hello”语句的例子来介绍如何编写JNI。
main.xml:
运行结果如下:
工程如下:
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 := JniTest
- LOCAL_SRC_FILES := JniTest.c
- include $(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");
- }