雅香小筑

喜欢技术,关注技术

导航

JNI技术基础(2)——从零开始编写JNI代码

 

书接上文: 《JNI技术基础(1)——从零开始编写JNI代码

 

2.编译源程序HelloWorld.java并生成HelloWorld.class

 

3.生成头文件HelloWorld.h

     在Linux控制台输入命令:javah –jni HelloWorld 生成HelloWorld.h头文件

//HelloWorld.h

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class HelloWorld */

#ifndef _Included_HelloWorld
#define _Included_HelloWorld
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     HelloWorld
 * Method:    print
 * Signature: ()V
 */
JNIEXPORT void JNICALL Java_HelloWorld_print
  (JNIEnv *, jobject);

#ifdef __cplusplus
}
#endif
#endif

       这个头文件中便告诉了我们需要用C/C++实现的函数的原型,即

              JNIEXPORT void JNICALL Java_HelloWorld_print ( JNIEnv * env, jobject obj)

       我们只需要按照这种格式完成其函数体的实现即可,函数名的格式:

              Java_类名_函数名

       参数env代表java虚拟机环境,Java传过来的参数和c有很大的不同,需要调用JVM提供的接口来转换成C类型的,就是通过调用env方法来完成转换的。
       参数obj代表调用的对象,相当于c++的this。当c函数需要改变调用对象成员变量时,可以通过操作这个对象来完成。


4.实现C/C++函数

       这块儿有点偷懒,并没有逐个字母去敲,而是通过拷贝头文件的方法,然后删除头文件中的无用信息,填充函数体的方法来创建HelloWorld.c文件,主要是因为JNI函数名都比较复杂,害怕疏忽敲错某个字符,或者少实现了某个函数,见谅。

//HelloWorld.c

#include <jni.h>
#include <stdio.h>
#include "HelloWorld.h"

/*
 * Class:     HelloWorld
 * Method:    print
 * Signature: ()V
 */

JNIEXPORT void JNICALL Java_HelloWorld_print (JNIEnv *, jobject){
    printf("Hello World!\n");
}

由上面可以看出,函数体的实现和普通的C/C++函数完全相同,不同的只是函数原型的格式。
需要注意的一点就是记得添加2个头文件:jni.h和HelloWorld.h 。

 

5.编译生成库文件

        gcc  -I/usr/lib/jvm/java-1.5.0-sun-1.5.0.22/include/linux/
                -I/usr/lib/jvm/java-1.5.0-sun-1.5.0.22/include/
                -fPIC  -shared  -o libHelloWorld.so  HelloWorld.c
       在编译的时候需要注意的就是记得加上java的两个路径,该路径根据你的java环境的实际安装路径而设置,其余的和编译普通的动态库方法相同。

第一个红色方框中圈出了我们经常范的一个错误,就是没有填写JNI函数的两个形参,虽然我们这里用不到它们,但是也必须写上,否则无法通过编译。

//HelloWorld.c

#include <jni.h>
#include <stdio.h>
#include "HelloWorld.h"

/*
 * Class:     HelloWorld
 * Method:    print
 * Signature: ()V
 */

JNIEXPORT void JNICALL Java_HelloWorld_print (JNIEnv *env, jobject obj){
    printf("Hello World!\n");
}

  

6.运行Java程序

方框圈出了两个经常范的错误,第一个错误产生的原因是找不到刚刚生成的C/C++动态库,需要手动指定库的路径,当然也可以把该库拷贝到系统库文件目录中。

手动指定动态库路径的方法是使用参数 -Djava.library.path

        java –Djava.library.path = "."  HelloWorld

第二个错误产生的原因是使用参数时,"="的前面或者后面使用了空格,去掉等号前后的空格即可。

 

OK,大功告成,终于看到久违的HelloWorld!

 

7.附录:一个简单的例程

<1>. MyTools.java

//MyTools.java


class MyTools{

    private native int myAdd(int x, int y);
    private native int mySub(int x, int y);

    public static void main(String[] args){

        int a = 5;
        int b = 7;
        int c = new MyTools().myAdd(a, b);
        int d = new MyTools().mySub(a, b);
        System.out.println(a + " + " + b + " = " + c);
        System.out.println(a + " - " + b + " = " + d);
    }

    static{
        System.loadLibrary("MyTools");
    }

}

  

<2>. 自动生成MyTools.h头文件

//MyTools.h

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class MyTools */

#ifndef _Included_MyTools
#define _Included_MyTools
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     MyTools
 * Method:    myAdd
 * Signature: (II)I
 */
JNIEXPORT jint JNICALL Java_MyTools_myAdd
  (JNIEnv *, jobject, jint, jint);

/*
 * Class:     MyTools
 * Method:    mySub
 * Signature: (II)I
 */
JNIEXPORT jint JNICALL Java_MyTools_mySub
  (JNIEnv *, jobject, jint, jint);

#ifdef __cplusplus
}
#endif
#endif

  

<3>. 函数体的C/C++实现myTools.c

//MyTools.c


#include <jni.h>
#include "MyTools.h"


/*
 * Class:     MyTools
 * Method:    myAdd
 * Signature: (II)I
 */
JNIEXPORT jint JNICALL Java_MyTools_myAdd (JNIEnv *env, jobject obj, jint x, jint y){
    return (x + y);
}

/*
 * Class:     MyTools
 * Method:    mySub
 * Signature: (II)I
 */
JNIEXPORT jint JNICALL Java_MyTools_mySub (JNIEnv *env, jobject obj, jint x, jint y){
    return (x - y);
}

运行结果:

 

说明:

     1.大多数情况下,JNI都是在Android开发中使用,本文的目的是使用最简单的语言描述出JNI最基本、最简单的使用流程,所以并没有使用Android框架,Android框架中的一大堆东西会阻挡我们的视线,无法专注于对JNI本身的了解,而Android下JNI的使用流程和Java是基本相同的,后面会有专门的篇幅介绍Android下JNI编程。
     2.此处实现了使用JNI传递简单的参数x和y。

     3.通过上面两个例子,相信大家已经可以写出自己的简单的 JNI 应用,但是整个 JNI 系统相当复杂,尤其是参数和返回值的传递方面,后面会有专门篇幅详细介绍。

 

< end >

 

 

posted on 2014-09-16 23:34  雅香小筑  阅读(841)  评论(0编辑  收藏  举报