本文是《Java Native Interface Programmer's Guide and Specification》的读书笔记

Java Native Interface可以让编程人员在Java里调用其他语言编写的方法,来弥补Java运行效率低下的缺点;

jni可以用来做什么?:

  • 在Java应用中使用本地编程语言如c/c++编写的代码;
  • 将Java虚拟机融入到由C/c++编写的的应用中;
  • 实现一个Java虚拟机;
  • 从技术层面理解语言的互操作性,特别是垃圾收集机制和多线程实现机制;

jni支持两种的本地代码:本地链接库和本地应用;

  • 使用jni在本地链接库中编写可以让Java应用调用的本地方法(native methods),Java应用可以像调用Java编写的方法一样调用本地方法(本地方法是由其他语言编写的,如C/C++);
  • jni允许调用接口(invocation interface),在本地应用中嵌入Java虚拟机(java virtual mathine),本地应用可以调用实现了Java虚拟机的本地链接库,然后就可以调用接口来执行Java编写的组件;

使用jni的步骤为:

  1. 创建一个java Class(如HelloWorld.java),在里面声明本地方法;
  2. 使用javac 命令编译源文件(HelloWorld.java),这个命令会在当前目录下生成一个HelloWorld.class文件;
  3. 使用命令:javah -jni 来生成一个C的头文件(HelloWord.h),头文件里包含本地方法实现的原型;
  4. 用C语言编写本地方法的实现(HelloWorld.c);
  5. 使用宿主环境的C编译器和链接器将本地方法的实现的文件(HelloWorld.c)编译为本地链接库(HelloWorld.dll 或者libHelloWorld.so);
  6. 运行HelloWorld的程序,类文件(HelloWorld.class)和链接库(HelloWorld.dll或者libHelloWorld.so)都会在运行时加载;

下面以一个HelloWorld的例子来掌握使用jni编程的步骤:

本地方法的声明:

class HelloWorld {
/**本地方法的声明与普通的Java方法的声明区别就是多一个native的修饰
**/
private native void print();//本地方法
private void doSomething(){};//普通Java方法
public static void main(String[] args) {
new HelloWorld().print();
}
static {
/**HelloWorld是链接库的名字,为了保证这个方法可以成功调用,必须创建好这个链接库(HelloWorld.dll在Win32里,libHelloWorld.so在Solaris中)***/
System.loadLibrary("HelloWorld");
}
}

native关键字表明这个方法是由其他语言实现的,但这个native方法被调用前,实现本地方法的本地链接库必须先被加载;如上面代码所示,在static区域加载这个本地链接库。Java虚拟机会在调用任何方法(本地方法或普通方法)前,自动初始化静态区域,这样就可以保证在调用本地方法时,链接库文件已经加载进来了。

编译原文件:使用命令

javac HelloWorld.java

这个命令执行成功后,会在当前目录下生成一个HelloWorld.class 文件;

生成头文件:使用命令:

 javah -jni HelloWorld

这个命令执行成功后,会生成一个HelloWorld.h的文件,头文件里最重要的就是本地方法的原型,也就是在用其他语言实现的的文件里,这个本地方法是以怎样一个形式出现的。上面的print()本地方法在头文件里的形式为:

JNIEXPORT void JNICALL
Java_HelloWorld_print (JNIEnv *, jobject);

JNIEXPORT和JNICALL是两个宏变量,用来保证可以从本地链接库导出这个本地方法并且C编译器会为这个本地方法生成正确的调用代码(怎样调用这个本地方法),函数名Java_HelloWorld_print,Java表明这是一个Java调用的方法,HelloWorld是原来的对象的名字,print是在Java里声明的方法的名字.我们发现本地方法的实现会包含两个参数,而我们声明本地方法是没有参数的,每一个本地方法的实现的第一个参数就是JNIEnv接口指针(包含所有的本地方法的入口指针),第二个参数是原来对象的引用(HelloWorld对象)类似于C++中的this指针;

实现本地方法:在HelloWorld.c中实现本地方法:

#include <jni.h> //这个头文件里,包含本地代码调用JNI方法的信息,在实现的文件里必须包含这个头文件
#include <stdio.h>//C语言头文件,需要使用里面提供的方法时才需要包含进来
#include "HelloWorld.h"//包含有本地方法原型,实现时,也必须包含进来
JNIEXPORT void JNICALL
Java_HelloWorld_print(JNIEnv *env, jobject obj)
{
printf("Hello World!\n");
return;
}

在编写完本地方法的实现后,就要将HelloWorld.c编译成本地链接库了,不同的操作系统下生成本地链接库的命令也不一样,在Solaris中,可以使用下面的命令生成本地链接库

cc -G -I/java/include -I/java/include/solaris  HelloWorld.c -o libHelloWorld.so

参数G表明让C编译器生成链接库,而不是一般的Solaris文件;

在Win32中,可以使用下面的命令让C++编译器生成动态链接库(命令需要写在同一行里面):

cl -Ic:\java\include -Ic:\java\include\win32
-MD -LD HelloWorld.c -FeHelloWorld.dll

最后就可以运行HelloWorld应用了,需要注意的是,你必须将链接库的路径让Java虚拟机可以搜索到,不然会找不到链接库,导致程序出错

Java Native Interface 编程系列一的更多相关文章

  1. Java Native Interface 六JNI中的异常

    本文是<The Java Native Interface Programmer's Guide and Specification>读书笔记 在这里只讨论调用JNI方法可能会出现的异常, ...

  2. Java Native Interface 二 JNI中对Java基本类型和引用类型的处理

    本文是<The Java Native Interface Programmer's Guide and Specification>读书笔记 Java编程里会使用到两种类型:基本类型(如 ...

  3. 【详解】JNI(Java Native Interface)(一)

    前言: 一提到JNI,多数编程者会下意识地感受到一种无法言喻的恐惧.它给人的第一感觉就是"难",因为它不是单纯地在JVM环境内操作Java代码,而是跳出虚拟机与其他编程语言进行交互 ...

  4. Java Native Interface Specification Contents 翻译

    https://docs.oracle.com/en/java/javase/12/docs/specs/jni/index.html Google翻译 第1章:简介 本章介绍Java Native ...

  5. JNI(Java Native Interface)

    一.JNI(Java Native Interface)        1.什么是JNI:               JNI(Java Native Interface):java本地开发接口   ...

  6. +Java中的native关键字浅析(Java+Native+Interface)++

    JNI是Java Native Interface的 缩写.从Java 1.1开始,Java Native Interface (JNI)标准成为java平台的一部分,它允许Java代码和其他语言写的 ...

  7. [ 高并发]Java高并发编程系列第二篇--线程同步

    高并发,听起来高大上的一个词汇,在身处于互联网潮的社会大趋势下,高并发赋予了更多的传奇色彩.首先,我们可以看到很多招聘中,会提到有高并发项目者优先.高并发,意味着,你的前雇主,有很大的业务层面的需求, ...

  8. Java Native Interface 五 JNI里的多线程与JNI方法的注册

    本文是<The Java Native Interface Programmer's Guide and Specification>读书笔记 JNI里的多线程 在本地方法里写有关多线程的 ...

  9. Java Native Interface 四--JNI中引用类型

    本文是<The Java Native Interface Programmer's Guide and Specification>读书笔记 JNI支持将类实例和数组类型(如jobjec ...

随机推荐

  1. iOS Run_time

    Runtime是想要做好iOS开发,或者说是真正的深刻的掌握OC这门语言所必需理解的东西.最近在学习Runtime,有自己的一些心得,整理如下,一为 查阅方便二为 或许能给他人一些启发,三为 希望得到 ...

  2. Excel到底最多可以有多少行

    平时我们使用的Excel到底最多可以有多少行? 答案:1048576,如图在红框中输入此数字即可看见.

  3. mvc SelectList selected失效的解决方法

    ViewBag.QuestionnaireType = new SelectList(questionCollectionTypeList, "CodeID", "Cod ...

  4. JS系列——Linq to js使用小结

    前言:前面几篇介绍了下C#基础技术中的几个:反射.特性.泛型.序列化.扩展方法.Linq to Xml等,本来还有两三个知识点没有写完,比如委托.多线程.异步等,后面会陆续将它们补起来,以便作为一套完 ...

  5. 通过接口实现JAVA和.NET互调用-JNInterface

    使用C#编程多年,也十分感激微软在语言架构.语法糖.编辑器等方面给自己带来的便利.但因为最近工作中有接触到JAVA,渐渐地发现的确像大家说的那样,JAVA的生态很好,要找点什么几乎都有现成的,于是自然 ...

  6. Windows Live Writer 2012 on .net 4

    众所周知,强大的live套件不再更新,止步于2012,所以安装的时候需要.net 3.5的运行时,这很讨厌,在新系统里面是.net 4运行时的,.net运行时的设计者脑袋张虫,新的居然不默认支持旧的程 ...

  7. Nhibernate Query By Criteria 条件查询

    HQL运算符 QBC运算符 含义 = Restrictions.eq() 等于equal <> Restrictions.ne() 不等于not equal > Restrictio ...

  8. Android四大组件之—— 使用服务进行后台操作

    什么是服务 服务是一个没有可视化界面的组件,它可以在后台长期运行并进行各种操作. 服务的创建 我们只需要继承Service类并实现相应的方法即可创建服务 要想启动服务,还得在AndroidManife ...

  9. Lua和C++交互详细总结

    转自:http://cn.cocos2d-x.org/tutorial/show?id=1474 一.Lua堆栈 要理解Lua和C++交互,首先要理解Lua堆栈. 简单来说,Lua和C/C++语言通信 ...

  10. 动态树之LCT(link-cut tree)讲解

    动态树是一类要求维护森林的连通性的题的总称,这类问题要求维护某个点到根的某些数据,支持树的切分,合并,以及对子树的某些操作.其中解决这一问题的某些简化版(不包括对子树的操作)的基础数据结构就是LCT( ...