本文是《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. 第26章 java进制操作

    java进制操作 1.二进制 二进制只有0和1,逢二进一 二进制多用在计算机中,来自计算机硬件的开关闭合 2.位运算 分别讲解: 2.1.按位与 & 两位全为1,结果才为1 0&0=0 ...

  2. Sublime中Markdown的安装与使用

    摘要:为什么用它,因为用markdown写出来的东西很好看,展示下:isujin.com(差不多就是这个样子啦,好看不?) 网页版Markdown编辑器有: 简书 jianshu.com等 客户端Ma ...

  3. Angular快速入门篇

    简介 AngularJS 是一个为动态WEB应用设计的结构框架,提供给大家一种新的开发应用方式,这种方式可以让你扩展HTML的语法,以弥补在构建动态WEB应用时静态文本的不足,从而在web应用程序中使 ...

  4. jQuery下拉框扩展和美化插件Chosen

    Chosen 是一个支持jquery的select下拉框美化插件,它能让丑陋的.很长的select选择框变的更好看.更方便.不仅如此,它更扩展了select,增加了自动筛选的功能.它可对列表进行分组, ...

  5. 在intellj idea下用sbt的坑

    version : SBT 0.13.7 intellij 14 新建SBT项目以后无法编译,总是显示 Can not resolve symble stackoverflow 给出了暂时的解决办法. ...

  6. 20145315&20145307《信息安全系统设计基础》实验五

    20145315&20145307<信息安全系统设计基础>实验五 北京电子科技学院(BESTI) 实 验 报 告 课程:信息安全系统设计基础 班级:1453 1452 姓名:陈俊达 ...

  7. Microsoft.VisualStudio.TestTools.UnitTesting 命名空间

      类 说明 AfterAssemblyCleanupEventArgs 为 AfterAssemblyCleanup 事件提供数据. AfterAssemblyInitializeEventArgs ...

  8. javascript数据类型判断

    javascript基本数据类型:原始类型  和  引用类型 原始类型(简单数据类型):String,Number,Boolean,Undefined,Null,Symbol(ES6新增) 引用类型( ...

  9. 【Python】[IO编程]文件读写,StringIO和BytesIO,操作文件和目录,序列化

    IO在计算机中指Input/Output,也就是输入和输出. 1.文件读写,1,读文件[使用Python内置函数,open,传入文件名标示符] >>> f = open('/User ...

  10. [React] 多组件生命周期转换关系

    前段时间一直在基于React做开发,最近得空做一些总结,防止以后踩坑. 言归正传,React生命周期是React组件运行的基础,本文主要是归纳多组件平行.嵌套时,生命周期转换关系. 生命周期 Reac ...