前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >JNI--c/c++调用Java方法

JNI--c/c++调用Java方法

作者头像
aruba
发布2020-07-03 13:59:24
3.8K0
发布2020-07-03 13:59:24
举报
文章被收录于专栏:android技术android技术
上次说到c/c++调用Java的变量,同样的c/c++也可以调用Java的方法
1.c/c++native方法调用Java非静态方法
首先看下我们Java的类
代码语言:javascript
复制
package com.aruba.jniapplication;

import java.util.Random;

public class JniDemo2 {

    static {
        System.load("C:\\Users\\tyqhc\\source\\repos\\JniApplication\\x64\\Debug\\JniApplication.dll");
    }

    public native void callRandom();

    private int getRandom(int bound) {
        return new Random().nextInt(bound);
    }

    public static void main(String[] args) {
    }

}
我们在c/c++代码中调用Java的getRandom方法,按照之前的套路,我们直接开始写c++代码

head文件

代码语言:javascript
复制
/* DO NOT EDIT THIS FILE - it is machine generated */
#include "jni.h"
/* Header for class com_aruba_jniapplication_JniDemo */

#ifndef _Included_com_aruba_jniapplication_JniDemo
#define _Included_com_aruba_jniapplication_JniDemo
#ifdef __cplusplus
extern "C" {
#endif

JNIEXPORT void JNICALL Java_com_aruba_jniapplication_JniDemo2_callRandom
(JNIEnv*, jobject);

#ifdef __cplusplus
}
#endif
#endif

cpp文件

代码语言:javascript
复制
#include "my_jni2.h"
#include<string>

using namespace std;

JNIEXPORT void JNICALL Java_com_aruba_jniapplication_JniDemo2_callRandom
(JNIEnv* env, jobject jobj) {
    //获取jclass
    jclass jclz = env->GetObjectClass(jobj);

    //三个参数:1:对应类的jclass 2:方法名 3:方法签名
    jmethodID jmid = env->GetMethodID(jclz,"getRandom","(I)I");
    //根据返回值,调用相应方法,我这边返回值是int,第三个参数为可变参数,就是调用方法需要传入的参数
    jint int_random = env->CallIntMethod(jobj, jmid,500);

    //打印
    printf("%d",int_random);
}
这边先获取jclass,再通过jclass获取jmethodID ,然后根据jmethodID调用Java的方法,最后打印输出。GetMethodID方法需要方法签名,下面介绍获取签名的方法,build下AS工程,找到我们需要反编译的class文件,如下图
在命令行使用javap命令
方法对应的descriptor就是方法签名,其实括号内就是参数签名,括号右边是返回值参数签名
编译c++后,再执行我们的Java程序
代码语言:javascript
复制
    public static void main(String[] args) {
        JniDemo2 jniDemo2 = new JniDemo2();
        jniDemo2.callRandom();
    }
Java方法被成功调用了,一般c/c++调用Java方法用于使用一些Java封装好的方法,而c/c++库函数又没有提供,自己手写又比较复杂,此时调用Java的方法会非常便利
2.c/c++静态native方法调用Java静态方法
我们在JniDemo2.java类中新增下面两个方法
代码语言:javascript
复制
    public static native void callUUID();

    private static String getUUID() {
        return UUID.randomUUID().toString();
    }
下面是c++代码
代码语言:javascript
复制
JNIEXPORT void JNICALL Java_com_aruba_jniapplication_JniDemo2_callUUID
(JNIEnv*, jclass);
代码语言:javascript
复制
JNIEXPORT void JNICALL Java_com_aruba_jniapplication_JniDemo2_callUUID
(JNIEnv* env, jclass jclz) {
    //获取静态方法的jmethodID
    jmethodID jmid = env->GetStaticMethodID(jclz, "getUUID", "()Ljava/lang/String;");
    //调用静态Java方法
    jstring jstr = (jstring)env->CallStaticObjectMethod(jclz, jmid);

    //将Java方法返回值转换为c++string对象
    string uuid = env->GetStringUTFChars(jstr, NULL);

    //创建一个文件
    string file_name = uuid + ".txt";
    string file_path = "C://Users//tyqhc//Desktop//" + file_name;
    FILE* f = fopen(file_path.c_str(),"w");
    fprintf(f,"hello Java %s",uuid.c_str());
    fclose(f);
}
这边调用了Java的getUUID方法,并创建一个以UUID命名的txt文件,内容是hello Java 加上UUID,编译后执行Java代码
代码语言:javascript
复制
package com.aruba.jniapplication;

import java.util.Random;
import java.util.UUID;

public class JniDemo2 {

    static {
        System.load("C:\\Users\\tyqhc\\source\\repos\\JniApplication\\x64\\Debug\\JniApplication.dll");
    }

    public native void callRandom();

    private int getRandom(int bound) {
        return new Random().nextInt(bound);
    }

    public static native void callUUID();

    private static String getUUID() {
        return UUID.randomUUID().toString();
    }

    public static void main(String[] args) {
        JniDemo2 jniDemo2 = new JniDemo2();
        jniDemo2.callRandom();

        jniDemo2.callUUID();
    }

}
我们的桌面生成了文件
内容是:
3.c/c++调用Java对象的构造方法
首先我们创建一个Java类:HelloC
代码语言:javascript
复制
public class HelloC {
    public HelloC() {
    }

    private long giveC() {
        return 1234;
    }
}
我们在JniDemo2.java中新增native方法,然后编写c++代码
代码语言:javascript
复制
/*
 * Class:     com_aruba_jniapplication_JniDemo2
 * Method:    createHelloC
 * Signature: ()Lcom/aruba/jniapplication/HelloC;
 */
JNIEXPORT jobject JNICALL Java_com_aruba_jniapplication_JniDemo2_createHelloC
  (JNIEnv *, jobject);
代码语言:javascript
复制
JNIEXPORT jobject JNICALL Java_com_aruba_jniapplication_JniDemo2_createHelloC
(JNIEnv* env, jobject jobj) {
    //首先找到我们需要实例化的class,通过包名加类名
    jclass hello_class = env->FindClass("com/aruba/jniapplication/HelloC");

    //获取构造方法jmethodID,所有构造方法名都是<init>
    jmethodID jmid = env->GetMethodID(hello_class, "<init>", "()V");
    //实例化
    jobject hello = env->NewObject(hello_class,jmid);

    //调用hello的giveC方法
    jmid = env->GetMethodID(hello_class, "giveC", "()J");
    jlong data = env->CallLongMethod(hello,jmid);
    printf("\n %ld",data);

    return hello;
}
所有构造方法名都是<init>,还调用了HelloC的giveC方法,编译后执行Java代码
代码语言:javascript
复制
    public static void main(String[] args) {
        JniDemo2 jniDemo2 = new JniDemo2();
        jniDemo2.callRandom();

       HelloC helloC = jniDemo2.createHelloC();
    }
成功的实例化了一个HelloC对象
总结:c/c++调用Java方法流程:根据jclass获取jmethodID->根据具体返回值调用相应的callXXXMethod方法或者构造方法的话调用NewObject方法
本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 上次说到c/c++调用Java的变量,同样的c/c++也可以调用Java的方法
  • 1.c/c++native方法调用Java非静态方法
    • 首先看下我们Java的类
      • 我们在c/c++代码中调用Java的getRandom方法,按照之前的套路,我们直接开始写c++代码
        • 这边先获取jclass,再通过jclass获取jmethodID ,然后根据jmethodID调用Java的方法,最后打印输出。GetMethodID方法需要方法签名,下面介绍获取签名的方法,build下AS工程,找到我们需要反编译的class文件,如下图
          • 在命令行使用javap命令
            • 方法对应的descriptor就是方法签名,其实括号内就是参数签名,括号右边是返回值参数签名
              • 编译c++后,再执行我们的Java程序
                • Java方法被成功调用了,一般c/c++调用Java方法用于使用一些Java封装好的方法,而c/c++库函数又没有提供,自己手写又比较复杂,此时调用Java的方法会非常便利
                • 2.c/c++静态native方法调用Java静态方法
                  • 我们在JniDemo2.java类中新增下面两个方法
                    • 下面是c++代码
                      • 这边调用了Java的getUUID方法,并创建一个以UUID命名的txt文件,内容是hello Java 加上UUID,编译后执行Java代码
                        • 我们的桌面生成了文件
                          • 内容是:
                          • 3.c/c++调用Java对象的构造方法
                            • 首先我们创建一个Java类:HelloC
                              • 我们在JniDemo2.java中新增native方法,然后编写c++代码
                                • 所有构造方法名都是<init>,还调用了HelloC的giveC方法,编译后执行Java代码
                                  • 成功的实例化了一个HelloC对象
                                  • 总结:c/c++调用Java方法流程:根据jclass获取jmethodID->根据具体返回值调用相应的callXXXMethod方法或者构造方法的话调用NewObject方法
                                  领券
                                  问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档