先看個(gè)命令:javap(獲取方法的簽名啊研,其實(shí)不用這個(gè)命令也行,自己應(yīng)該也看的出來(lái)方法的簽名整葡,只是介紹個(gè)工具)
javap命令操作的是class文件
調(diào)用java非靜態(tài)方法
以這段代碼為例:
package com.example.huozhenpeng.myapplication;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.TextView;
public class MainActivity extends AppCompatActivity {
// Used to load the 'native-lib' library on application startup.
static {
System.loadLibrary("native-lib");
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
TextView tv = (TextView) findViewById(R.id.sample_text);
tv.setText("");
}
public String getJavaMethod(int max)
{
return "helloworld";
}
}
hhh:myapplication huozhenpeng$ javap -s -p MainActivity.class
Compiled from "MainActivity.java"
public class com.example.huozhenpeng.myapplication.MainActivity extends android.support.v7.app.AppCompatActivity {
public com.example.huozhenpeng.myapplication.MainActivity();
descriptor: ()V
protected void onCreate(android.os.Bundle);
descriptor: (Landroid/os/Bundle;)V
public java.lang.String getJavaMethod(int);
descriptor: (I)Ljava/lang/String;
static {};
descriptor: ()V
}
示例
package com.example.huozhenpeng.myapplication;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.widget.TextView;
public class MainActivity extends AppCompatActivity {
// Used to load the 'native-lib' library on application startup.
static {
System.loadLibrary("native-lib");
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
getJavaMethod();
}
public native void getJavaMethod();
public String addMethod(int max)
{
return max+"helloworld";
}
}
extern "C"
JNIEXPORT void JNICALL
Java_com_example_huozhenpeng_myapplication_MainActivity_getJavaMethod(JNIEnv *env,
jobject instance) {
jclass jcs=env->GetObjectClass(instance);
jmethodID jmd=env->GetMethodID(jcs,"addMethod","(I)Ljava/lang/String;");//L表示的是對(duì)象類(lèi)型
jobject job=env->CallObjectMethod(instance,jmd,200);
jstring js= (jstring) job;
const char *value=env->GetStringUTFChars(js,NULL);
printf("%s\d",value);
}
找不到輸出c++的地方虏缸,直接debug查看吧:
調(diào)用java靜態(tài)方法
這樣一個(gè)場(chǎng)景:在java中生成uuid(c語(yǔ)言生成uuid比較復(fù)雜),在c中調(diào)用java中的方法獲取uuid蚁堤,生成一個(gè)文件uuid.txt,并向文件寫(xiě)入helloworld醉者。
試了很多次,androidstudio始終fopen文件失敗披诗,找不出原因撬即,操作文件的功能先放下。
private native void generateFileByC();
private static String getUUID()
{
return UUID.randomUUID().toString();
}
extern "C"
JNIEXPORT void JNICALL
Java_com_example_huozhenpeng_myapplication_MainActivity_generateFileByC(JNIEnv *env,
jobject instance) {
jclass jcl=env->GetObjectClass(instance);
jmethodID jmd=env->GetStaticMethodID(jcl,"getUUID","()Ljava/lang/String;");
jstring jsg= (jstring) env->CallStaticObjectMethod(jcl, jmd);
const char *p =env->GetStringUTFChars(jsg,NULL);
char s[100];
sprintf(s,"/storage/emulated/0/%s.txt",p);
printf("%s\n",s);
}
jni操作java構(gòu)造方法
例如:c語(yǔ)言中沒(méi)有Date對(duì)象呈队,如果我們要獲取當(dāng)前時(shí)間就比較麻煩剥槐,所以直接在jni中獲取java的Date然后調(diào)用Date的getTime()方法。
TextView textView= (TextView) findViewById(R.id.sample_text);
SimpleDateFormat simpleDateFormat=new SimpleDateFormat("yyyy-MM-dd");
Date date=new Date(testJniCustructor());
textView.setText("當(dāng)前時(shí)間:"+simpleDateFormat.format(date));
private native long testJniCustructor();
extern "C"
JNIEXPORT jlong JNICALL
Java_com_example_huozhenpeng_myapplication_MainActivity_testJniCustructor(JNIEnv *env,
jobject instance) {
jclass jcs=env->FindClass("java/util/Date");
jmethodID jmd=env->GetMethodID(jcs,"<init>","()V");//構(gòu)造函數(shù)的方法名都是<init>
//實(shí)例化Date
jobject job=env->NewObject(jcs,jmd);
//得到getTime的methodId
jmethodID jme=env->GetMethodID(jcs,"getTime","()J");//long的簽名是J不是L
jlong jl=env->CallLongMethod(job,jme);
return jl;
}
測(cè)試結(jié)果:
操作String
關(guān)于String亂碼
在java層定義String宪摧,傳遞給jni
setString("好好學(xué)習(xí)");
private native void setString(String str);
extern "C"
JNIEXPORT void JNICALL
Java_com_example_huozhenpeng_myapplication_MainActivity_setString(JNIEnv *env, jobject instance,
jstring str_) {
const char *str = env->GetStringUTFChars(str_, NULL);
env->ReleaseStringUTFChars(str_, str);
}
沒(méi)有演示出來(lái)亂碼粒竖,尷尬了。有亂碼的百度下啊几于,基本上windows的話(huà)用的是WideCharToMultiByte蕊苗,其他的貌似需要借助三方。
在jni定義字符串沿彭,傳給java
extern "C"
JNIEXPORT jstring JNICALL
Java_com_example_huozhenpeng_myapplication_MainActivity_getStringFromC(JNIEnv *env,
jobject instance) {
const char *p="哈哈哈哈";
return env->NewStringUTF(p);
}
private native String getStringFromC();
TextView textView= (TextView) findViewById(R.id.sample_text);
textView.setText(getStringFromC());
測(cè)試結(jié)果:
沒(méi)有亂碼朽砰,尷尬。
如果有亂碼的話(huà)可以借助java層的String類(lèi)帶charset的構(gòu)造方法實(shí)現(xiàn)。
看下GetStringUTFChars的第三個(gè)參數(shù)的作用:
當(dāng)從JNI函數(shù)GetStringUTFChars函數(shù)中返回得到字符串B時(shí)瞧柔,如果B是原始字符串java.lang.String的一份拷貝漆弄,則isCopy 被賦值為JNI_TRUE。如果B是和原始字符串指向的是JVM中的同一份數(shù)據(jù)造锅,則isCopy 被賦值為JNI_FALSE撼唾。 當(dāng)isCopy 為JNI_FALSE時(shí),本地代碼絕不能修改字符串的內(nèi)容备绽,否則JVM中的原始字符串也會(huì)被修改券坞,這會(huì)打破Java語(yǔ)言中字符串不可變的規(guī)則