JNI&NDK编程小结及建议

damk3990 8年前
   <h2>前言</h2>    <p>由于网上关于JNI/NDK相关的知识点介绍的比较零散而且不具备参照性,所以写了这篇JNI/NDK笔记,便于作为随时查阅的工具类型的文章,本文主要的介绍了在平时项目中常用的命令、JNI数据类型、签名等,便于查阅相关资料。文末相关参考资料比较适合刚接触或者不熟悉Android NDK开发的朋友参阅。</p>    <h2>常用命令</h2>    <h3>javac 编译java源文件生成.class文件</h3>    <p>由于JNI对应的头文件由javah工具根据对应的.class文件生成,所以在进行JNI编程之前,写好Java代码后需要先编译,在使用javah生成对应的头文件</p>    <h3>javah -jni自动生成头文件</h3>    <p>举例说明:</p>    <ul>     <li> <p>生成普通的JNI头文件</p> <pre>  <code class="language-java">javah -classpath path -jni -d outputdirpath com.mrljdx.JavaNativeCode  </code></pre> </li>     <li> <p>在Java函数中包含Android相关的参数代码,则需要在classpath中添加android.jar包的绝对路径地址</p> <pre>  <code class="language-java">javah -classpath path:$ANDROID_HOME/path/android.jar -jni -d outputdirpath com.mrljdx.JavaNativeCodeWithAndroid  </code></pre> </li>    </ul>    <h3>javap -s -p 查看函数签名</h3>    <p>-s: 显示签名(只显示public类型的签名) -p:显示所有函数、成员变量的签名</p>    <p>举例说明:</p>    <pre>  <code class="language-java">javap -classpath pacakage_path_dir -s -p com.mrljdx.JavaCode  </code></pre>    <h2>JNI数据类型和类型签名</h2>    <h3>数据类型</h3>    <p>JNI的数据类型包括: <strong>基本类型</strong> 和 <strong>引用类型</strong> 。这一点和Java的语言特性一致,基本类型包括jboolean、jchar、jint、jlong、jbyte、jshort、jfloat、jdouble、void,与Java类型的对应关系如下:</p>    <table>     <thead>      <tr>       <th>JNI类型</th>       <th>Java类型</th>       <th>描述</th>      </tr>     </thead>     <tbody>      <tr>       <td>jboolean</td>       <td>boolean</td>       <td>无符号8位整型</td>      </tr>      <tr>       <td>jbyte</td>       <td>byte</td>       <td>有符号8位整型</td>      </tr>      <tr>       <td>jchar</td>       <td>char</td>       <td>无符号16位整型</td>      </tr>      <tr>       <td>jshort</td>       <td>short</td>       <td>有符号16位整型</td>      </tr>      <tr>       <td>jint</td>       <td>int</td>       <td>32位整型</td>      </tr>      <tr>       <td>jlong</td>       <td>long</td>       <td>64位整型</td>      </tr>      <tr>       <td>jfloat</td>       <td>float</td>       <td>32位整型</td>      </tr>      <tr>       <td>jdouble</td>       <td>double</td>       <td>64位整型</td>      </tr>      <tr>       <td>void</td>       <td>void</td>       <td>无类型</td>      </tr>     </tbody>    </table>    <p>JNI中引用类型主要有类、对象和数组,这点也上符合Java的语法规范,对应的关系如下:</p>    <table>     <thead>      <tr>       <th>JNI 类型</th>       <th>Java引用类型</th>       <th>描述</th>      </tr>     </thead>     <tbody>      <tr>       <td>jobject</td>       <td>Object</td>       <td>Object类型</td>      </tr>      <tr>       <td>jclass</td>       <td>Class</td>       <td>Class类型</td>      </tr>      <tr>       <td>jstring</td>       <td>String</td>       <td>String类型</td>      </tr>      <tr>       <td>jobjectArray</td>       <td>Object[]</td>       <td>对象数组</td>      </tr>      <tr>       <td>jbooleanArray</td>       <td>boolean[]</td>       <td>boolean数组</td>      </tr>      <tr>       <td>jbyteArray</td>       <td>byte[]</td>       <td>byte数组</td>      </tr>      <tr>       <td>jcharArray</td>       <td>char[]</td>       <td>char数组</td>      </tr>      <tr>       <td>jshortArray</td>       <td>short[]</td>       <td>short数组</td>      </tr>      <tr>       <td>jintArray</td>       <td>int[]</td>       <td>int数组</td>      </tr>      <tr>       <td>jlongArray</td>       <td>long[]</td>       <td>long数组</td>      </tr>      <tr>       <td>jfloatArray</td>       <td>float[]</td>       <td>float数组</td>      </tr>      <tr>       <td>jdoubleArray</td>       <td>double[]</td>       <td>double数组</td>      </tr>      <tr>       <td>jthrowable</td>       <td>Throwable</td>       <td>Throwable</td>      </tr>     </tbody>    </table>    <h3>JNI类型签名</h3>    <p>JNI的类型签名标识了一个特定的Java类型,这个类型可以是类和方法,也可以是数据类型。</p>    <ul>     <li>类型签名<br> 类的签名采用”L+包名+类名+;”标识,包名中将 . 替换为 / 即可。<br> 比如String类的签名:<br> Ljava/lang/String;<br> 注意末尾的 ; 属于签名的一部分。<br> 再比如Android中Context类的签名:<br> Landroid/content/Context;</li>     <li>基本数据类型签名<br> 基本数据类型的签名采用一系列大写字母来标识,如下:</li>    </ul>    <table>     <thead>      <tr>       <th>Java类型</th>       <th>签名</th>      </tr>     </thead>     <tbody>      <tr>       <td>boolean</td>       <td>Z</td>      </tr>      <tr>       <td>byte</td>       <td>B</td>      </tr>      <tr>       <td>char</td>       <td>C</td>      </tr>      <tr>       <td>short</td>       <td>S</td>      </tr>      <tr>       <td>int</td>       <td>I</td>      </tr>      <tr>       <td>long</td>       <td>J</td>      </tr>      <tr>       <td>float</td>       <td>F</td>      </tr>      <tr>       <td>double</td>       <td>D</td>      </tr>      <tr>       <td>void</td>       <td>V</td>      </tr>     </tbody>    </table>    <p>可以发现除了 long 基本数据类型的签名为 J 之外其他的都比较容易辨识,估计是由于之前的类类型的签名开头为 L+包名+类名+; 设计者为了区分所以签名为 J</p>    <ul>     <li>数组的类型签名<br> 数组的类型签名比起类类型和基本数据类型的要稍微复杂一点,不过还是很好理解的。对于数组来说,它的签名为 [+类型签名 ,举例说明:<br> String[] 数组类型对应的签名:<br> [Ljava/lang/String;<br> 可以发现,就是在String的类签名前加了个 [<br> 同理基本数据类型签名int[]的签名:<br> [I<br> 注意这里基本类型后面是不带分号的。<br> 那么多维数组呢?可以类推,int[][] 的签名为 [[I ,而String[][]的签名为 [[Ljava/lang/String;</li>     <li>方法的签名<br> 在JNI中会经常需要在C/C++代码中调用Java的函数,这时候就会用到方法的签名。方法的签名为 (+参数类型签名+)+返回值类型签名 ,比如:<br> 方法: boolean login(String username,String password) 的方法签名如下:<br> (Ljava/lang/String;Ljava/lang/String)B 如果这里不理解的话,请再去看看之前关于基本类型,类类型的签名部分内容。</li>    </ul>    <p>小技巧:使用 类似于 javap -classpath pathdir -s -p com.sample.JavaCode 的 javap -s -p 命令也可以帮助查看一些类中各种方法和成员变量的签名。</p>    <h2>JNI相关命名解释</h2>    <ul>     <li>函数名的格式遵循规则: Java_包名_类名_方法名</li>     <li>JNIEXPORT、JNICALL、JNIEnv和jobject 都是JNI标准中所定义的类型或者宏</li>     <li>JNIEnv * : 指向JNI环境的指针,可以通过JNIEnv * 访问JNI提供的接口方法</li>     <li>JNIEXPORT、JNICALL:是jni.h中所定义的宏。</li>    </ul>    <p>注:JNIEnv * 可以简单的理解为Java和C/C++ 之间相互调用的桥梁,我们可以通过JNIEnv * 调用C/C++定义的方法,也可以在C/C++中通过JNIEnv * 来调用Java类中的方法。下面将会讲到C/C++中调用Java的方法,注意JNIEnv *的作用。</p>    <h2>在C/C++中调用Java方法</h2>    <p>首先说明一点,在Android开发过程中使用NDK主要是为了提高代码的安全性,有些游戏公司可能是为了方便利用已有的C/C++开源库来进行平台移植,其实在性能提升方面,NDK的作用并不是很明显。所以有时候一些在Java中实现起来非常简单的代码放在JNI里面做会显得吃力不讨好,所以干脆就直接在JNI中调用Java的方法,我们只把加密和验证的一些逻辑写到JNI层就行了。在JNI中调用Java方法流程如下:</p>    <ol>     <li>在Java中定义一个 <strong>静态</strong> 方法供JNI调用,注意要是静态的。</li>     <li>在JNI中利用env来调用Java中定义的静态方法</li>     <li>调用声明好的静态方法</li>    </ol>    <p>可能流程说的比较抽象,用代码简单说明一下:</p>    <ol>     <li> <p>定义静态方法:</p> <pre>  <code class="language-java">//对应包名:com.mrljdx.jni.HelloJNI  public static void helloJava() {          System.out.println("Hello JavaCode");  }  </code></pre> </li>     <li> <p>JNI声明静态方法:</p> <pre>  <code class="language-java">static void static_helloJava(JNIEnv *env){        jclass clazz = env->FindClass("com/mrljdx/jni/HelloJNI");  }  </code></pre> </li>     <li> <p>调用声明好的静态方法:</p> <pre>  <code class="language-java">static_helloJava(env);  </code></pre> </li>    </ol>    <h2>在AndroidStudio中NDK编程配置注意事项:</h2>    <ol>     <li> <p>在项目的 gradle.properties 中添加ndk支持:</p> <pre>  <code class="language-java">android.useDeprecatedNdk=true  </code></pre> </li>     <li> <p>配置 build.gradle 看代码注释:</p> <pre>  <code class="language-java">defaultConfig {           minSdkVersion 9          targetSdkVersion 23          versionCode 1           versionName "1.0"          //配置ndk 支持       ndk {                    //编译的so库名称 libsecurity.so            moduleName "security"                   //指定编译后的库支持的平台            abiFilters "armeabi", "mips", "x86", "armeabi-v7a"              //用于指定应用应该使用哪个标准库,此处添加c++库支持             stl "stlport_static"            }  }  </code></pre> </li>     <li> <p>在AndroidStudio中写JNI代码有一个比较爽的地方,就是Android.mk系统会在编译时自动帮你生成,你只需要配置build.gradle就行了。注意jni相关代码需要放在 src/main/jni 目录下。如果对gradle配置不了解可以参考我的博客:Gradle实战及学习建议</p> </li>    </ol>    <h2>小结</h2>    <p>在我们做产品的时候,应该考虑该用JNI&NDK的时候就用,一切出发点是基于用户的体验和数据安全,我觉得在以下几种情况下建议使用NDK:</p>    <ol>     <li>重用现有的代码,比如C/C++的代码在Android中的重用。</li>     <li>数据安全,比如将Http的请求加密和解密算法放在NDK中去实现,这样可以提高应用的安全。</li>     <li>提升性能,由于Android设备制造商在手机中给每个应用分配了可用的最大RAM,有时候为了性能考虑,可以通过Native代码向系统来“借”一些内存,尽量少的使用系统分配给应用的内存。(参考Infoq: <a href="/misc/goto?guid=4958986149242339462" rel="nofollow,noindex">Android内存优化</a> )</li>    </ol>    <p>来自: <a href="/misc/goto?guid=4959670892711548519" rel="nofollow">http://mrljdx.com/2016/04/16/JNI-NDK编程小结及建议/</a> </p>