`
nanapoleon
  • 浏览: 46816 次
  • 性别: Icon_minigender_1
  • 来自: 西安
社区版块
存档分类
最新评论

本地方法调用3

    博客分类:
  • java
阅读更多
    继续本地调用的相关知识。前面已经讲过了关于java如何调用下面的相关知识,包括如何适配变量,jni调用的方法规范,以及一些jni方法调用的注意事项等等。
    根据前面的介绍,我们可以发现,调用的过程,如果真的使用起来,并没有很顺手,因为仅仅只有参数的向下传递是绝对不够的。我们不仅要下层的代码访问上层的,还要来回调用彼此之间的方法,这样做才够cool嘛。而且才能灵活满足各种需求。就像佛经所说:能进能退,乃真正法器(形容能进能退,进退有度,赵云式的霸气),有木有?
    那先来介绍下访问实例域吧。在java程序中,有很多方法,会使用到类中的全局变量,比如下面的代码:
class Employee{
    double salary = 0;
    
    public void raiseSalary(double byPercent){
        salary += 1 + byPercent / 100;
    }
}

    很常见的方法吧?现在呢,我们要把这个raiseSalary变成一个本地方法。根据前面内容,javah可以生成一个如下方法
JNIEXPORT void JNICALL Java_Employee_raiseSalary(JNIEnv *, jobject, jdouble)

    看到这个方法,你能看出和先前讲得有什么不同么?什么?能进能退?我代表赵云,给你跪了。。。咳,我们看到,有两个不同处,第一个,salary变量是一个Employee的变量,并没有被通过参数传入到c层中。这个我们要稍后重点介绍。还有一个,看看第二个传入参数,可以看到,前面由于使用的是static静态方法,所以,传递进来的是jclass,而这里,我们看到传进来的是jobject。这个也很合理,因为静态方法是没有对象的嘛。
    好了,有java对象传下来,后面就很好想象了,对吧。不过,直接访问要求虚拟机暴露它们的内部数据布局,因此,JNI要求程序员通过调用特殊的JNI函数来获取和设置数据的值。这里我们要使用GetdoubleField和SetDoubleField函数,当然还有其他类型,一般语法是:
x = (*env)->GetXxxField(env, this_obj, fieldID);
(*env)->SetXxxField(env, this_obj, fieldID, x);

    这里,又产生了两个疑问,一个是fieldID是个什么东东?还有,GetObjectField方法返回的对象,我们用什么class定义它?
    先看看FieldID,它是一个特殊类型的值,fieldID表示结构中的一个域。使用GetFieldID函数来获得fieldID。
    jfieldID id_salary = (*env)->GetFieldID(env, class_Employee, "salary", "D");

    字符串"D"表示类型是double。这个后面介绍。
    感觉有点像纹理id有没有?JNI的设计者不想把数据域直接暴露在外,所以,他们不得不提供获取和设置数据域值的函数。为了使这些函数的开销最小化,从域名计算域ID(代价最大的一个步骤)被分解出来作为单一步骤。这样,如果你反复获取和设置一个特定的域,你计算域标示符的开销只有一次。
    第二点,如何得到一个class?jni有两种方式得到class:
   
jclass class_Employee = (*env)->GetObjectClass(env, this_obj);

    FindClass函数可以让你以字符串形式来指定类名(很奇怪的是使用/来代替包分割符号.)
   
jclass class_String = (*env)->FindClass(env, "java/lang/String");

    看起来有点像反射,是吧。(越往后看,其实jni的java层的method调用也是反射的形式,他们分别对应了反射的field和method)
    好了,大概介绍完了如何访问java的域,我们最后做个总结,把开始的方法完全转换成jni层的代码:
JNIEXPORT void JNICALL Java_Employee_raiseSalary(JNIEnv* env, jobject this_obj, jdouble byPercent){
    jclass class_Employee = (*env)->GetObjectClass(env, this_obj);
    jfieldID id_salary = (*env)->GetFiledID(env, class_Employee, "salary", "D");
    jdouble salary = (*env)->getDoubleField(env, this_obj, id_salary);
    salary += 1 + byPercent;
    (*env)->SetDoubleField(env, this_obj, id_salary, salary);
}
分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics