先看个命令:javap(获取方法的签名,其实不用这个命令也行,自己应该也看的出来方法的签名,只是介绍个工具)
javap命令操作的是class文件
调用java非静态方法
以这段代码为例:
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表示的是对象类型
jobject job=env->CallObjectMethod(instance,jmd,200);
jstring js= (jstring) job;
const char *value=env->GetStringUTFChars(js,NULL);
printf("%s\d",value);
}
找不到输出c++的地方,直接debug查看吧:
调用java静态方法
这样一个场景:在java中生成uuid(c语言生成uuid比较复杂),在c中调用java中的方法获取uuid,生成一个文件uuid.txt,并向文件写入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构造方法
例如:c语言中没有Date对象,如果我们要获取当前时间就比较麻烦,所以直接在jni中获取java的Date然后调用Date的getTime()方法。
TextView textView= (TextView) findViewById(R.id.sample_text);
SimpleDateFormat simpleDateFormat=new SimpleDateFormat("yyyy-MM-dd");
Date date=new Date(testJniCustructor());
textView.setText("当前时间:"+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,"","()V");//构造函数的方法名都是
//实例化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;
}
测试结果:
操作String
关于String乱码
在java层定义String,传递给jni
setString("好好学习");
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);
}
没有演示出来乱码,尴尬了。有乱码的百度下啊,基本上windows的话用的是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());
测试结果:
没有乱码,尴尬。
如果有乱码的话可以借助java层的String类带charset的构造方法实现。
看下GetStringUTFChars的第三个参数的作用:
当从JNI函数GetStringUTFChars函数中返回得到字符串B时,如果B是原始字符串java.lang.String的一份拷贝,则isCopy 被赋值为JNI_TRUE。如果B是和原始字符串指向的是JVM中的同一份数据,则isCopy 被赋值为JNI_FALSE。 当isCopy 为JNI_FALSE时,本地代码绝不能修改字符串的内容,否则JVM中的原始字符串也会被修改,这会打破Java语言中字符串不可变的规则