点击按钮返回Native获取的IMEI号并显示。
- 创建一个新的Android工程
- Build—>Make Project生成class文件
- javah生成头文件
- 写C文件
- NDK配置
- Build—>Make Project生成so文件
步骤1:创建一个新的Android工程,这个不是这篇文章的重点而且网上教程很多,不知道的小伙伴可以在百度上搜哦~~
不知道为什么我的把SDK中android-24下的android.jar包拖到jd-gui看发现没有appCompatActivity,而且我下载的最新的sdk\extras\里也没有android-support-v7-appcompat.jar包,知道的小伙伴拜托告诉我一下下。步骤2一直报无法访问android.support.v7.app.AppCompatActivity的错误。为了体验下一步,我偷了一下下懒,这里的MainActivity是继承了最原始的那种Activity。
为Button绑定一个OnclickListener,使得点击按钮就调用so的native方法getIMEI
public class MainActivity extends Activity {
Button button;
TextView textView;
static{
System.loadLibrary("imei-jni");
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
button = (Button) findViewById(R.id.button);
textView = (TextView)findViewById(R.id.text);
button.setOnClickListener(new MyClickListener());
}
class MyClickListener implements OnClickListener{
public void onClick(View v){
String imei = getIMEI();
textView.setText(imei);
}
}
public native String getIMEI();
}
步骤2:Build—>Make Project生成class文件。点击Build->MakeProject,会在app_path/build/intermediates/classes/debug下生成class文件,下一步需要使用到这些class文件。
步骤:3: javah生成头文件。在Terminal(View->Tool Windows->Terminal)进到 app_path\src\main的目录,然后输入下面命令:
javah -d jni -classpath SDK_android.jar;APP_classes activity
这个activity就是包含了native方法的activity。
结果会在main的jni目录生成头文件,如下图
那个MyClickListener.h的头文件其实没有什么用。
步骤4:写C文件
新建一个c文件,然后include步骤3生成的头文件。
#include <jni.h>
#include "com_example_a414_hellowworld_MainActivity.h"
JNIEXPORT jstring JNICALL Java_com_example_a414_hellowworld_MainActivity_getIMEI
(JNIEnv *env, jobject mContext){
if(mContext == 0){
return (*env)->NewStringUTF(env,"[+]Error : Context is 0");
}
jclass cls_context = (*env)->FindClass(env,"android/content/Context");
if(cls_context == 0){
return (*env)->NewStringUTF(env,"[+] Error: FindClass <android/content/Context> Error");
}
jmethodID getSystemService = (*env)->GetMethodID(env,cls_context,"getSystemService","(Ljava/lang/String;)Ljava/lang/Object;");
if(getSystemService == 0){
return (*env)->NewStringUTF(env,"[+] Error : GetMethodID failed");
}
jfieldID TELEPHONY_SERVICE = (*env)->GetStaticFieldID(env,cls_context,"TELEPHONY_SERVICE","Ljava/lang/String;");
if(TELEPHONY_SERVICE == 0){
return (*env)->NewStringUTF(env,"[+] Error : GetStaticFieldID failed");
}
jstring str = (jstring)(*env)->GetStaticObjectField(env,cls_context, TELEPHONY_SERVICE);
jobject telephonymanager = ((*env)->CallObjectMethod(env,mContext, getSystemService, str));
if(telephonymanager == 0){
return (*env)->NewStringUTF(env,"[+] Error: CallObjectMethod failed");
}
jclass cls_TelephoneManager = (*env)->FindClass(env, "android/telephony/TelephonyManager");
if(cls_TelephoneManager == 0){
return (*env)->NewStringUTF(env,"[+] Error: FindClass TelephoneManager failed");
}
jmethodID getDeviceId = ((*env)->GetMethodID(env,cls_TelephoneManager, "getDeviceId", "()Ljava/lang/String;"));
if(getDeviceId == 0){
return (*env)->NewStringUTF(env,"[+] Error: GetMethodID getDeviceID failed");
}
jobject DeviceID = (*env)->CallObjectMethod(env,telephonymanager,getDeviceId);
return (jstring)DeviceID;
}
听说Windows上有个什么bug,所以最好在当前目录下创建一个空的c文件,我这里创建的是util.c文件。
步骤5:配置NDK。为了能生成so文件,需要在几个文件中添加NDK的配置:
1.app目录下的build.gradle。defaultConfig里加上
ndk{
muduleName “module_name“
abiFilters “armeabi”,”armeabi-v7a”,”x86”
}
defaultConfig {
applicationId "com.example.a414.hellowworld"
minSdkVersion 9
targetSdkVersion 24
versionCode 1
versionName "1.0"
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
ndk {
moduleName "imei-jni"
abiFilters "armeabi", "armeabi-v7a", "x86"
}
}
2.在gradle.properties里添加android.useDeprecateNdk=true
步骤6:Build—>Make Project生成so文件
再点击一次Build—>Make Project,会在app_path/build/intermediates/ndk/debug/lib下生成so文件。
步骤7:
在MainActivity上右击run,程序就可以成功运行起来了。
然而此时,我的工程中还存在一个错误:
于是我把步骤6生成的so的文件夹放到app下的libs里,再修改app下的build.gradle文件如下:
apply plugin: 'com.android.application'
android {
compileSdkVersion 24
buildToolsVersion "24.0.2"
defaultConfig {
applicationId "com.example.a414.hellowworld"
minSdkVersion 9
targetSdkVersion 24
versionCode 1
versionName "1.0"
testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
}
sourceSets.main {
jni.srcDirs = []
jniLibs.srcDir 'src/main/libs'
}
task nativeLibsToJar(type: Zip, description: "create a jar archive of the native libs") {
destinationDir file("$projectDir/libs")
baseName "libimei-jni"
extension "jar"
from fileTree(dir: "libs", include: "**/*.so")
into "lib"
}
tasks.withType(JavaCompile) {
compileTask -> compileTask.dependsOn(nativeLibsToJar)
}
}
dependencies {
compile fileTree(dir: 'libs', include: ['*.jar'])
androidTestCompile('com.android.support.test.espresso:espresso-core:2.2.2', {
exclude group: 'com.android.support', module: 'support-annotations'
})
compile 'com.android.support:appcompat-v7:24.2.0'
testCompile 'junit:junit:4.12'
}
这个问题依然存在,但是我点运行依然能在模拟器上正常运行,这是为什么,有哪位小伙伴知道呀?