在android开发中,有时候需要编写一些C/C++代码,这时候就要用到JNI技术,我们需要将C/C++程序首先编译成so库,在java中通过native方法调用so库中的函数。
实现以上目的有三种方式:
1、单独编译so库文件,将它push到手机的system/lib目录下, 在java程序中通过loadLibrary加载so库。
2、使用NDK工具进行编译,需要配置NDK环境,然后通过Android Studio将其打包打APK中。
3、在Android源码环境中使用mm,so文件就能够打包到APK文件中,随着APK一起发布,而不是将so文件放到系统目录中。
如果是开发系统级的APK,就需要选择第三种方法,下面是第三种方法的具体实例。
APP编译
Android.mk
LOCAL_PATH := $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE_TAGS := tests
LOCAL_MULTILIB := 32
LOCAL_PROGUARD_ENABLED := disabled
LOCAL_SDK_VERSION := current
LOCAL_SRC_FILES := \
$(call all-java-files-under, src) \
$(call all-renderscript-files-under, src)
LOCAL_PACKAGE_NAME := MyJni
LOCAL_JNI_SHARED_LIBRARIES := libmy_jni
include $(BUILD_PACKAGE)
include $(call all-makefiles-under,$(LOCAL_PATH))
AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.app"
android:versionCode="1"
android:versionName="1.0" >
<uses-sdk android:minSdkVersion="18" android:targetSdkVersion="19"/>
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name=".MainActivity"
android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
</manifest>
JNI编译
jni/Android.mk
LOCAL_PATH:= $(call my-dir)
include $(CLEAR_VARS)
LOCAL_MODULE := libmy_jni
#需要特别注意,缺失会导致library “libc++.so” not found
LOCAL_SDK_VERSION := 14
LOCAL_MODULE_TAGS := optional
LOCAL_SRC_FILES := Addjni.c
LOCAL_CFLAGS += -Wall -Wextra -Werror -Wno-unused-parameter
DISABLE_DEXPREOPT:=true
LOCAL_NDK_STL_VARIANT := stlport_static
include $(BUILD_SHARED_LIBRARY)
jni/Application.mk
APP_STL := stlport_static
MainActivity.java
package com.example.app;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.TextView;
import android.app.Activity;
public class MainActivity extends Activity {
private TextView mSum;
private EditText mEditText1;
private EditText mEditText2;
private Button mButton;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mSum = (TextView) findViewById(R.id.sum);
mEditText1 = (EditText) findViewById(R.id.number1);
mEditText2 = (EditText) findViewById(R.id.number2);
mButton = (Button)findViewById(R.id.button_result);
mButton.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
String mNum1 = mEditText1.getText().toString();
String mNum2 = mEditText2.getText().toString();
if(mNum1.equals("") || mNum2.equals(""))
return;
android.util.Log.i("wwd","mNum1 = "+mNum1+", mNum2 = "+mNum2);
int add1 = Integer.parseInt(mNum1);
int add2 = Integer.parseInt(mNum2);
int summary = add(add1, add2);
mSum.setText("sum = " + summary);
}
});
}
//调用SO库中的函数
public native int add(int num1, int num2);
//这里加载SO库。
static{
System.loadLibrary("my_jni");
}
}
jni/Addjni.c
#include <stdio.h>
#include <jni.h>
#include "Addjni.h"
//这里就是我们调用C/C++中的功能模块,实际开发中可以调用其他的各种C类去实现功能
int addNumber(int num1, int num2){
return num1 + num2;
}
//MainActivity中调用的add函数,作用就是连接java层和C/C++层的API,格式是java+包名+类名+函数名
jint Java_com_example_app_MainActivity_add(JNIEnv* env, jobject thiz, jint num1, jint num2){
return addNumber(num1, num2);
}
jni/Addjni.h
#ifndef ADDJNI_H
#define ADDJNI_H
extern int add(int num1, int num2);
#endif
我们只需要在apk根目录执行mm编译Android.mk,jni/Android.mk就会执行,而且生成的so文件会自动打包到apk文件中。