Android NDK开发三:java和C\C++交互
1、定义native方法并加载动态链接库:
public class HelloJni extends Activity { @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); TextView tv = new TextView(this); tv.setText( stringFromJNI() ); setContentView(tv); } public native String stringFromJNI(); public native String unimplementedStringFromJNI(); static { System.loadLibrary("hello-jni"); } }
2、实现native方法:
#include <string.h> #include <jni.h> jstring Java_com_example_hellojni_HelloJni_stringFromJNI( JNIEnv* env, jobject thiz ) { return (*env)->NewStringUTF(env, "Hello from JNI !"); //in c
return env->NewStringUTF("Hello from JNI !"); //in c++
}
若要定义静态方法:
JNIEXPORT jstring JNICALL Java_com_example_hellojni_HelloJni_stringFromJNI (JNIEnv * env, jclass clazz);
头文件可以用javah工具生成:
进入命令行,切换到项目的bin目录下的classes下面,执行 javah -classpath . -jni 包.类名。
或者:cd 到src目录中,执行 javah 包.类名
3、数据
原始数据类型:
引用数据类型:
4、字符窜的操作:
1)新建字符串:
jstring javaString;
javaString = (*env)->NewStringUTF(env, "Hello World!");
2)把java string转换成c string
const jbyte* str; jboolean isCopy; str = (*env)->GetStringUTFChars(env, javaString, &isCopy); if (0 != str) { printf("Java string: %s", str); if (JNI_TRUE == isCopy) { printf("C string is a copy of the Java string."); } else { printf("C string points to actual string."); } (*env)->ReleaseStringUTFChars(env, javaString, str);
5、数组的操作:
1)新建java数组:
jintArray javaArray; javaArray = (*env)->NewIntArray(env, 10); if (0 != javaArray) { /* You can now use the array. */ }
2)获取数组的值:
jint nativeArray[10];
//将java array 复制到 c array (*env)->GetIntArrayRegion(env, javaArray, 0, 10, nativeArray);
//将 c array 还原城 java array
(*env)->SetIntArrayRegion(env, javaArray, 0, 10, nativeArray);
获取数组指针:
jint* nativeDirectArray; jboolean isCopy; nativeDirectArray = (*env)->GetIntArrayElements(env, javaArray, &isCopy);
//...
(*env)->ReleaseIntArrayElements(env, javaArray, nativeDirectArray, 0);
6、C\C++获取java成员变量
public class JavaClass { /** Instance field */ private String instanceField = "Instance Field"; /** Static field */ private static String staticField = "Static Field"; ... }
1)获取Field ID
jclass clazz;
clazz = (*env)->GetObjectClass(env, instance);
jfieldID instanceFieldId; instanceFieldId = (*env)->GetFieldID(env, clazz, "instanceField", "Ljava/lang/String;");
jfieldID staticFieldId; staticFieldId = (*env)->GetStaticFieldID(env, clazz, "staticField", "Ljava/lang/String;");
2)获取Static Field
jstring staticField;
staticField = (*env)->GetStaticObjectField(env, clazz, staticFieldId);
7、C\C++调用java函数
public class JavaClass { /** * Instance method. */ private String instanceMethod() { return "Instance Method"; } /** * Static method. */ private static String staticMethod() { return "Static Method"; } ... }
1)获取Method ID:
成员函数:
jmethodID instanceMethodId; instanceMethodId = (*env)->GetMethodID(env, clazz, "instanceMethod", "()Ljava/lang/String;");
静态函数:
jmethodID staticMethodId; staticMethodId = (*env)->GetStaticMethodID(env, clazz, "staticMethod", "()Ljava/lang/String;");
2)调用:
jstring instanceMethodResult; instanceMethodResult = (*env)->CallStringMethod(env, instance, instanceMethodId);
变量和函数的描述符:
javap工具可以提取从编译过的class files 中提取这些描述符:
javap –classpath bin/classes –p –s com.example.hellojni.HelloJni
8、处理异常:
public class JavaClass { /** * Throwing method. */ private void throwingMethod() throws NullPointerException { throw new NullPointerException("Null pointer"); } /** * Access methods native method. */ private native void accessMethods(); }
jthrowable ex; ... (*env)->CallVoidMethod(env, instance, throwingMethodId); ex = (*env)->ExceptionOccurred(env); if (0 != ex) { (*env)->ExceptionClear(env); /* Exception handler. */ }
从c\c++抛异常:
jclass clazz; ... clazz = (*env)->FindClass(env, "java/lang/NullPointerException"); if (0 ! = clazz) { (*env)->ThrowNew(env, clazz, "Exception message."); }
9、本地和全局变量:
1)本地变量在函数结束时自动释放内存:
jclass clazz; clazz = (*env)->FindClass(env, "java/lang/String");
2)新建全局变量:
jclass localClazz; jclass globalClazz; ... localClazz = (*env)->FindClass(env, "java/lang/String"); globalClazz = (*env)->NewGlobalRef(env, localClazz); ... (*env)->DeleteLocalRef(env, localClazz);
3)释放全局变量:
(*env)->DeleteGlobalRef(env, globalClazz);
4)弱全局引用:
弱全局引用可以被内存自动回收:
jclass weakGlobalClazz; weakGlobalClazz = (*env)->NewWeakGlobalRef(env, localClazz); if (JNI_FALSE == (*env)->IsSameObject(env, weakGlobalClazz, NULL)) { /* Object is still live and can be used. */ } else { /* Object is garbage collected and cannot be used. */ } (*env)->DeleteWeakGlobalRef(env, weakGlobalClazz);
10、线程:
synchronized(obj) { /* Synchronized thread-safe code block. */ }
if (JNI_OK == (*env)->MonitorEnter(env, obj)) { /* Error handling. */ } /* Synchronized thread-safe code block. */ if (JNI_OK == (*env)->MonitorExit(env, obj)) { /* Error handling. */ }
native线程不能和java直接通信,必须先绑定到javaVM
JavaVM* cachedJvm; ... JNIEnv* env; ... /* Attach the current thread to virtual machine. */ (*cachedJvm)->AttachCurrentThread(cachedJvm, &env, NULL); /* Thread can communicate with the Java application using the JNIEnv interface. */ /* Detach the current thread from virtual machine. */ (*cachedJvm)->DetachCurrentThread(cachedJvm);
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· 阿里巴巴 QwQ-32B真的超越了 DeepSeek R-1吗?
· 【译】Visual Studio 中新的强大生产力特性
· 【设计模式】告别冗长if-else语句:使用策略模式优化代码结构
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义