如何从 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类