如何从 JNI android 调用 java 类及其方法?

Posted

技术标签:

【中文标题】如何从 JNI android 调用 java 类及其方法?【英文标题】:How to call java class and its method from JNI android? 【发布时间】:2018-07-09 09:45:30 【问题描述】:

我想在我的 cpp 类中调用内部 java 类。

这是我的简单 java 类(活动)

 public class MainActivity extends AppCompatActivity 

    static 

        System.loadLibrary("myLibrary");
    

    @Override
    protected void onCreate(Bundle savedInstanceState) 
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        Toast.makeText(this, "" + stringFromJNI(), Toast.LENGTH_SHORT).show();
        checkAES();
        checkRSA();
        invokeclass();
        Log.d("Gajanand", "onCreate: "+invokeclass());
    
   public native String stringFromJNI();

    public native int checkRSA();

    public native void checkAES();

    public native int invokeclass();

    public static class bar 
        public static int timesTen(int input)
            return input * 10;
        
    


我的 cpp 类如下所示

#include<jni.h>
#include <string>

#include <android/log.h>
#include <openssl/rsa.h>
#include <openssl/aes.h>


extern "C"
JNIEXPORT jstring  JNICALL
Java_com_manvish_bwssb_Activities_MainActivity_stringFromJNI(
        JNIEnv *env,
        jobject /* this */) 
    std::string hello = "Hello from C++";
    return env->NewStringUTF(hello.c_str());

#define nr_bits 2048

extern "C"
JNIEXPORT int JNICALL
  Java_com_manvish_bwssb_Activities_MainActivity_checkRSA(JNIEnv *env, jobject /* this */) 
    RSA *rsa = RSA_generate_key(nr_bits, 65537, NULL, NULL);
    return 0;


static const unsigned char key[] = 
        0x00, 0x11, 0x22, 0x33, 0x44, 0x55, 0x66, 0x77,
        0x88, 0x99, 0xaa, 0xbb, 0xcc, 0xdd, 0xee, 0xff,
        0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07,
        0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f
;

extern "C"
void Java_com_manvish_bwssb_Activities_MainActivity_checkAES(JNIEnv *env, jobject /* this */) 

    unsigned char text[] = "Kishan of Nanda";
    unsigned char out[80];
    unsigned char decout[80];

    AES_KEY wctx;

    AES_set_encrypt_key(key, 128, &wctx);
    AES_encrypt(text, out, &wctx);
    printf("encryp data = %s\n", out);
    __android_log_print(ANDROID_LOG_DEBUG, "Gajanand", "Encrypted data: %s\n", out);

    AES_set_decrypt_key(key, 128, &wctx);
    AES_decrypt(out, decout, &wctx);

    printf(" Decrypted o/p: %s \n", decout);
    __android_log_print(ANDROID_LOG_DEBUG, "Gajanand", "Decrypted data: %s\n", decout);

extern "C"
JNIEXPORT int JNICALL
 Java_com_manvish_bwssb_Activities_MainActivity_invokeclass(JNIEnv *env)

    jclass myClass = env->FindClass("bar");
    jmethodID mid = env->GetStaticMethodID(myClass, "timesTen", "(I)I");
    jint hundred = env->CallStaticIntMethod(myClass, mid, (jint)10);
    return hundred;

我班上发生的事情是 checkRSA() 和 checkAES() 重复调用。我不明白为什么只有这两个方法重复调用。但是 invokeclass() 方法根本没有调用。这是调用 java 类的正确方法还是有其他方法。?我的应用程序中的另一个问题是构建应用程序需要超过 3 分钟。帮助我解决这两个问题。

【问题讨论】:

【参考方案1】:

您没有指定 bar 类的路径,只需输入您的包路径,将 . 替换为 / 并且由于它是内部类 - 您应该使用 YourClass$YourInnerClass 访问它。

所以在你的情况下是这样的:

jclass myClass = env->FindClass("com/mavish/bwssb/Activities/MainActivity$bar");

另外请注意,env->FindClass 和类似的调用确实会导致异常,因此请仔细阅读Exception Handling in JNI

导致该问题的另一个常见原因是缩小器,因此请检查您的 Proguard 规则。确保他们不删除任何本机调用或您的静态 Bar 类(如果它没有在 java 中的任何地方调用 - 它可能会在 minify 阶段被删除)

How to keep native methods

How to keep inner classes

关于您非常长的构建时间 - 我会说问题太宽泛了,因为有很多可能的问题会使构建时间变长。我建议您修改根项目文件夹中的 gradle.properties,添加这些参数:

org.gradle.parallel=true 
org.gradle.daemon=true 
org.gradle.jvmargs=-Xmx8192m -XX:MaxPermSize=4096 -XX:+HeapDumpOnOutOfMemoryError 

【讨论】:

以上是关于如何从 JNI android 调用 java 类及其方法?的主要内容,如果未能解决你的问题,请参考以下文章

Android在C ++ JNI代码中从另一个活动类调用Java函数

蓝牙与 Android 上的 Qt。通过抽象类上的jni调用java类

Android JNI之C/C++层调用JAVA

Android Studio NDK开发-JNI调用Java方法

Android JNI之C/C++层调用JAVA

如何在 JNI 中使用自定义类类型参数调用 Java 函数