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

时间:2018-01-30 13:26:32

标签: java android c++ android-ndk java-native-interface

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

这是我的简单java类(Activity)

 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 个答案:

答案 0 :(得分:0)

您没有指定条形码类的路径,只需放置包裹路径,将.替换为/,因为它是内部类 - 您应该使用YourClass$YourInnerClass访问它。

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

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

另请注意,env-&gt; 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