Android中JNI高级应用 - 本地C代码中创建Java对象及本地JNI对象的保存
2012-03-28 12:27
746 查看
++++++++++++++++++++++++++++++++++++++++++
本文系本站原创,欢迎转载! 转载请注明出处:
/article/1354959.html
++++++++++++++++++++++++++++++++++++++++++
本地C代码中创建Java对象
创建Java域的对象就是创建Java类的实例,再调用Java类的构造方法。
以Bitmap的构建为例,Bitmap中并没有Java对象创建的代码及外部能访问的构造方法,所以它的实例化必然是在JNI的c中实现的。
BitmapFactory.java中提供了得到Bitmap的方法,时序简化为:
BitmapFactory.java ->BitmapFactory.cpp -> GraphicsJNI::createBitmap() [graphics.cpp]
GraphicsJNI::createBitmap()[graphics.cpp]的实现:
[cpp] view plaincopy
jobjectGraphicsJNI::createBitmap(JNIEnv* env, SkBitmap* bitmap, bool isMutable,
jbyteArray ninepatch, intdensity)
{
SkASSERT(bitmap != NULL);
SkASSERT(NULL != bitmap->pixelRef());
jobject obj =env->AllocObject(gBitmap_class);
if (obj) {
env->CallVoidMethod(obj,gBitmap_constructorMethodID,
(jint)bitmap,isMutable, ninepatch, density);
if (hasException(env)) {
obj = NULL;
}
}
return obj;
}
而gBitmap_class的得到是通过:
[cpp] view plaincopy
jclass c =env->FindClass("android/graphics/Bitmap");
gBitmap_class =(jclass)env->NewGlobalRef(c);
//gBitmap_constructorMethodID是Bitmap的构造方法(方法名用”<init>”)的jmethodID:
gBitmap_constructorMethodID= env->GetMethodID(gBitmap_class, "<init>",
"(IZ[BI)V");
总结一下,c中如何访问Java对象的属性:
1) 通过JNIEnv::FindClass()找到对应的jclass;
2) 通过JNIEnv::GetMethodID()找到类的构造方法的jfieldID;
3) 通过JNIEnv::AllocObject创建该类的对象;
4) 通过JNIEnv::CallVoidMethod()调用Java对象的构造方法。
本地JNI对象的保存
c域中某次被调用生成的对象,在其他函数调用时是不可见的,虽然可以设置全局变量但那不是好的解决方式,Android中通常是在Java域中定义一个int型的变量,在c域生成对象的地方,与这个Java域的变量关联,在别的使用到的地方,在从这个变量中取值。
以JNICameraContext为例来说明:
JNICameraContext是android_hardware_camera.cpp中定义的类型,并会在cpp中生成对象,与Java中android.hardware.Camera的mNativeContext关联。
在注册native函数之前,c中就已经把Java域中的属性的jfieldID得到了。通过下列方法
[cpp] view plaincopy
jclass clazz = env->FindClass("android/hardware/Camera ");
jfieldID field = env->GetFieldID(clazz, " mNativeContext","I");
如果执行成功,把field保存到上面图中的fileds变量的context:jfieldID中。
生成cpp对象时,通过JNIEnv::SetIntField()设置为Java对象的属性
[cpp] view plaincopy
static void android_hardware_Camera_native_setup(JNIEnv *env, jobjectthiz,
jobject weak_this, jintcameraId)
{
// …
// We use a weak reference sothe Camera object can be garbage collected.
// The reference is only used asa proxy for callbacks.
sp<JNICameraContext>context = new JNICameraContext(env, weak_this, clazz, camera);
// …
// 该处通过context.get()得到context对象的地址,保存到了Java中的mNativeContext属性里
env->SetIntField(thiz,fields.context, (int)context.get());
}
而要使用时,又通过JNIEnv::GetIntField()获取Java对象的属性,并转化为JNICameraContext类型:
[cpp] view plaincopy
JNICameraContext* context =reinterpret_cast<JNICameraContext*>(env->GetIntField(thiz, fields.context));
if (context != NULL) {
// …
}
总结一下,c++中生成的对象如何保存和使用:
1) 通过JNIEnv::FindClass()找到对应的jclass;
2) 通过JNIEnv::GetFieldID()找到类中属性的jfieldID;
3) 某个调用过程中,生成cpp对象时,通过JNIEnv::SetIntField()设置为Java对象的属性;
4) 另外的调用过程中,通过JNIEnv::GetIntField()获取Java对象的属性,再转化为真实的对象类型。
JNI完全详解:http://mindprod.com/jgloss/jni.html
++++++++++++++++++++++++++++++++++++++++++
本文系本站原创,欢迎转载! 转载请注明出处:
/article/1354959.html
++++++++++++++++++++++++++++++++++++++++++
本文系本站原创,欢迎转载! 转载请注明出处:
/article/1354959.html
++++++++++++++++++++++++++++++++++++++++++
本地C代码中创建Java对象
创建Java域的对象就是创建Java类的实例,再调用Java类的构造方法。
以Bitmap的构建为例,Bitmap中并没有Java对象创建的代码及外部能访问的构造方法,所以它的实例化必然是在JNI的c中实现的。
BitmapFactory.java中提供了得到Bitmap的方法,时序简化为:
BitmapFactory.java ->BitmapFactory.cpp -> GraphicsJNI::createBitmap() [graphics.cpp]
GraphicsJNI::createBitmap()[graphics.cpp]的实现:
[cpp] view plaincopy
jobjectGraphicsJNI::createBitmap(JNIEnv* env, SkBitmap* bitmap, bool isMutable,
jbyteArray ninepatch, intdensity)
{
SkASSERT(bitmap != NULL);
SkASSERT(NULL != bitmap->pixelRef());
jobject obj =env->AllocObject(gBitmap_class);
if (obj) {
env->CallVoidMethod(obj,gBitmap_constructorMethodID,
(jint)bitmap,isMutable, ninepatch, density);
if (hasException(env)) {
obj = NULL;
}
}
return obj;
}
而gBitmap_class的得到是通过:
[cpp] view plaincopy
jclass c =env->FindClass("android/graphics/Bitmap");
gBitmap_class =(jclass)env->NewGlobalRef(c);
//gBitmap_constructorMethodID是Bitmap的构造方法(方法名用”<init>”)的jmethodID:
gBitmap_constructorMethodID= env->GetMethodID(gBitmap_class, "<init>",
"(IZ[BI)V");
总结一下,c中如何访问Java对象的属性:
1) 通过JNIEnv::FindClass()找到对应的jclass;
2) 通过JNIEnv::GetMethodID()找到类的构造方法的jfieldID;
3) 通过JNIEnv::AllocObject创建该类的对象;
4) 通过JNIEnv::CallVoidMethod()调用Java对象的构造方法。
本地JNI对象的保存
c域中某次被调用生成的对象,在其他函数调用时是不可见的,虽然可以设置全局变量但那不是好的解决方式,Android中通常是在Java域中定义一个int型的变量,在c域生成对象的地方,与这个Java域的变量关联,在别的使用到的地方,在从这个变量中取值。
以JNICameraContext为例来说明:
JNICameraContext是android_hardware_camera.cpp中定义的类型,并会在cpp中生成对象,与Java中android.hardware.Camera的mNativeContext关联。
在注册native函数之前,c中就已经把Java域中的属性的jfieldID得到了。通过下列方法
[cpp] view plaincopy
jclass clazz = env->FindClass("android/hardware/Camera ");
jfieldID field = env->GetFieldID(clazz, " mNativeContext","I");
如果执行成功,把field保存到上面图中的fileds变量的context:jfieldID中。
生成cpp对象时,通过JNIEnv::SetIntField()设置为Java对象的属性
[cpp] view plaincopy
static void android_hardware_Camera_native_setup(JNIEnv *env, jobjectthiz,
jobject weak_this, jintcameraId)
{
// …
// We use a weak reference sothe Camera object can be garbage collected.
// The reference is only used asa proxy for callbacks.
sp<JNICameraContext>context = new JNICameraContext(env, weak_this, clazz, camera);
// …
// 该处通过context.get()得到context对象的地址,保存到了Java中的mNativeContext属性里
env->SetIntField(thiz,fields.context, (int)context.get());
}
而要使用时,又通过JNIEnv::GetIntField()获取Java对象的属性,并转化为JNICameraContext类型:
[cpp] view plaincopy
JNICameraContext* context =reinterpret_cast<JNICameraContext*>(env->GetIntField(thiz, fields.context));
if (context != NULL) {
// …
}
总结一下,c++中生成的对象如何保存和使用:
1) 通过JNIEnv::FindClass()找到对应的jclass;
2) 通过JNIEnv::GetFieldID()找到类中属性的jfieldID;
3) 某个调用过程中,生成cpp对象时,通过JNIEnv::SetIntField()设置为Java对象的属性;
4) 另外的调用过程中,通过JNIEnv::GetIntField()获取Java对象的属性,再转化为真实的对象类型。
JNI完全详解:http://mindprod.com/jgloss/jni.html
++++++++++++++++++++++++++++++++++++++++++
本文系本站原创,欢迎转载! 转载请注明出处:
/article/1354959.html
++++++++++++++++++++++++++++++++++++++++++
相关文章推荐
- Android中JNI高级应用 - 本地C代码中创建Java对象及本地JNI对象的保存
- Android中JNI高级应用 - 本地C代码中创建Java对象及本地JNI对象的保存 .
- Android中JNI高级应用 - 本地C代码中创建Java对象及本地JNI对象的保存
- Android中JNI高级应用 - 本地C代码中创建Java对象及本地JNI对象的保存
- JNI学习笔记:(1)开篇(2)本地代码访问Java代码 (3)本地方法取得Java属性/调用java方法 (4)本地代码创建Java对象(包括javaString) (5) 本地方法处理java数组
- JNI学习笔记4——本地代码创建Java对象(包括javaString)
- JNI学习笔记4——本地代码创建Java对象(包括javaString)
- jni c本地代码回调java (多线程中,创建java对象时报空指针)未解决
- IntelliJ IDEA平台下JNI编程(五)—本地C代码创建Java对象及引用
- JNI中如何在本地代码中生成和操作Java的对象和字符串,并编写应用实例
- 深入了解android平台的jni---本地多线程调用java代码
- Android Java代码中设置style属性--创建ProgressBar对象
- Android 在Java代码中设置style属性--使用代码创建ProgressBar对象(反射机制)
- JNI学习笔记4——本地代码创建Java对象(包括javaString)
- Android: JNI本地函数控制Java端代码
- Android 在Java代码中设置style属性--使用代码创建ProgressBar对象
- Android 在Java代码中设置style属性--使用代码创建ProgressBar对象
- JNI学习笔记(四)JNI中本地语言创建Java对象并且访问具体方法(附例子)
- 深入了解android平台的jni的原理以及本地多线程调用java代码
- Android JNI 使用的数据结构JNINativeMethod详解 ||建立Android SDK下的JNI、JAVA应用完整步骤---Android JAVA调用C++代码