转载请注明出处:http://blog.csdn.net/xyang81/article/details/42066665java
JNI中的基本类型和Java中的基本类型都是一一对应的,接下来先看一下JNI的基本类型定义:编程
基本类型很容易理解,就是对C/C++中的基本类型用typedef从新定义了一个新的名字,在JNI中能够直接访问。 JNI把Java中的全部对象看成一个C指针传递到本地方法中,这个指针指向JVM中的内部数据结构,而内部的数据结构在内存中的存储方式是不可见的。只能从JNIEnv指针指向的函数表中选择合适的JNI函数来操做JVM中的数据结构。第三章的示例中,访问java.lang.String对应的JNI类型jstring时,没有像访问基本数据类型同样直接使用,由于它在Java是一个引用类型,因此在本地代码中只能经过GetStringUTFChars这样的JNI函数来访问字符串的内容。安全
示例解析:数据结构
1> 访问字符串函数
sayHello函数接收一个jstring类型的参数text,但jstring类型是指向JVM内部的一个字符串,和C风格的字符串类型char*不一样,因此在JNI中不能通把jstring看成普通C字符串同样来使用,必须使用合适的JNI函数来访问JVM内部的字符串数据结构。编码
GetStringUTFChars(env, j_str, &isCopy) 参数说明:spa
env:JNIEnv函数表指针.net
j_str:jstring类型(Java传递给本地代码的字符串指针)指针
isCopy:取值JNI_TRUE和JNI_FALSE,若是值为JNI_TRUE,表示返回JVM内部源字符串的一份拷贝,并为新产生的字符串分配内存空间。若是值为JNI_FALSE,表示返回JVM内部源字符串的指针,意味着能够经过指针修改源字符串的内容,不推荐这么作,由于这样作就打破了Java字符串不能修改的规定。但咱们在开发当中,并不关心这个值是多少,一般状况下这个参数填NULL便可。code
由于Java默认使用Unicode编码,而C/C++默认使用UTF编码,因此在本地代码中操做字符串的时候,必须使用合适的JNI函数把jstring转换成C风格的字符串。JNI支持字符串在Unicode和UTF-8两种编码之间转换,GetStringUTFChars能够把一个jstring指针(指向JVM内部的Unicode字符序列)转换成一个UTF-8格式的C字符串。在上例中sayHello函数中咱们经过GetStringUTFChars正确取得了JVM内部的字符串内容。
2> 异常检查
调用完GetStringUTFChars以后不要忘记安全检查,由于JVM须要为新诞生的字符串分配内存空间,当内存空间不够分配的时候,会致使调用失败,失败后GetStringUTFChars会返回NULL,并抛出一个OutOfMemoryError异常。JNI的异常和Java中的异常处理流程是不同的,Java遇到异常若是没有捕获,程序会当即中止运行。而JNI遇到未决的异常不会改变程序的运行流程,也就是程序会继续往下走,这样后面针对这个字符串的全部操做都是很是危险的,所以,咱们须要用return语句跳事后面的代码,并当即结束当前方法。
3> 释放字符串
在调用GetStringUTFChars函数从JVM内部获取一个字符串以后,JVM内部会分配一块新的内存,用于存储源字符串的拷贝,以便本地代码访问和修改。即然有内存分配,用完以后立刻释放是一个编程的好习惯。经过调用ReleaseStringUTFChars函数通知JVM这块内存已经不使用了,你能够清除了。注意:这两个函数是配对使用的,用了GetXXX就必须调用ReleaseXXX,并且这两个函数的命名也有规律,除了前面的Get和Release以外,后面的都同样。
4> 建立字符串
经过调用NewStringUTF函数,会构建一个新的java.lang.String字符串对象。这个新建立的字符串会自动转换成Java支持的Unicode编码。若是JVM不能为构造java.lang.String分配足够的内存,NewStringUTF会抛出一个OutOfMemoryError异常,并返回NULL。在这个例子中咱们没必要检查它的返回值,若是NewStringUTF建立java.lang.String失败,OutOfMemoryError这个异常会被在Sample.main方法中抛出。若是NewStringUTF建立java.lang.String成功,则返回一个JNI引用,这个引用指向新建立的java.lang.String对象。