第八节(c\c++调用java代码)

先看个命令: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";
    }
}
第八节(c\c++调用java代码)_第1张图片
image.png
第八节(c\c++调用java代码)_第2张图片
image.png
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查看吧:


第八节(c\c++调用java代码)_第3张图片
image.png

调用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);

}
第八节(c\c++调用java代码)_第4张图片
image.png

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;

}

测试结果:

第八节(c\c++调用java代码)_第5张图片
image.png

操作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);
}
第八节(c\c++调用java代码)_第6张图片
image.png

没有演示出来乱码,尴尬了。有乱码的百度下啊,基本上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());

测试结果:

第八节(c\c++调用java代码)_第7张图片
image.png

没有乱码,尴尬。
如果有乱码的话可以借助java层的String类带charset的构造方法实现。

看下GetStringUTFChars的第三个参数的作用:
当从JNI函数GetStringUTFChars函数中返回得到字符串B时,如果B是原始字符串java.lang.String的一份拷贝,则isCopy 被赋值为JNI_TRUE。如果B是和原始字符串指向的是JVM中的同一份数据,则isCopy 被赋值为JNI_FALSE。 当isCopy 为JNI_FALSE时,本地代码绝不能修改字符串的内容,否则JVM中的原始字符串也会被修改,这会打破Java语言中字符串不可变的规则

你可能感兴趣的:(第八节(c\c++调用java代码))