JNI编程实现(Linux)

JNIJava Native Interface的缩写,是Java平台的本地调用,从Java1.1就成为了Java标准的一部分,它允许Java代码和其它语言的代码进行互相调用,只要调用约定支持即可,尤其和C/C++的互相调用。

虽然使用Java与本地编译的代码进行交互,会丧失平台的可移植性,但是在特定情况下,这些问题是可以接受的,如:

1.使用一些旧的库

2.需要操作系统交互

3.提高程序的性能

一、jni介绍

Java是通过定义native方法,然后用其它语言实现该方法,最后在Java运行时,动态地加载该方法实现,通过调用native的方法,进而实现Java的本地调用。

1.实现架构

JVM封装了各种操作系统的差异性,提供了jni技术,使得开发中可以通过Java程序调用到操作系统的函数,进而与其它技术进行交互。下图是Linux平台jni的调用流程。Java应用程序通过jni接口调用动态链接库*.so,来实现jni的功能。

2.类型映射

Java基本数据类型与C语言基本数据类型的对应

3.常用方法简介

1) GetStringUTFLength

以字节为单位返回字符串的UTF-8长度

    // jsize (JNICALL *GetStringUTFLength)(JNIEnv *env, jstring str)
    int len = (*env)->GetStringUTFLength(env, str);

2) GetStringUTFChars

返回指向字符串的UTF-8字符数组的指针。该数组在被ReleaseStringUTFChars()释放前将一直有效

    // const char* (JNICALL *GetStringUTFChars)(JNIEnv *env, jstring str, jboolean *isCopy)
    const char *buf = (*env)->GetStringUTFChars(env, str, NULL);

isCopyJNI_FALSE,不要修改返回值,不然将改变java.lang.String的不可变语义。 一般会把isCopy设为NULL,不关心Java VM对返回的指针是否直接指向java.lang.String的内容

3) ReleaseStringUTFChars

通知虚拟机平台相关代码无需再访问utfutf参数是一个指针,可利用GetStringUTFChars()获得

    // void (JNICALL *ReleaseStringUTFChars)(JNIEnv *env, jstring str, const char* chars)
    (*env)->ReleaseStringUTFChars(env, str, buf);

4) NewStringUTF

利用UTF-8字符数组构造新java.lang.String对象

    // jstring (JNICALL *NewStringUTF)(JNIEnv *env, const char *utf)
    (*env)->NewStringUTF(env, "hello");

更多实用方法,请参考jni.h

二、jni实现步骤

下面介绍jni的具体实现步骤,主要是通过Java程序调用C方法,跑通整儿jni的调用流程。

1.编写java类

编写JavaHello类,定义一个native的本地方法

public class Hello {
    public native static String sayHello(String name);

    static {
        System.load("你的*.so的绝对路径");
    }

    public static void main(String[] args) {
        Hello hello = new Hello();
        String ret = hello.sayHello("kelvin");
        System.out.println(ret);
    }
}

2.编译java类

使用javac命令进行编译

# javac Hello.java

3.生成本地文件*.h

这是关键的一步,主要是生成本地方法签名,依赖的是上一步的class文件,

# javah -jni Hello

如果你的java源文件有包名,在生成*.sh的时候,也要带包名转化的路径,即用classpath指定包所在的路径,不然在最后调用时,会报错:UnsatisfiledLinkError

// java源文件包名
package kelvin.Java.dynamicso;

// 编译时指定classpath
# javah -classpath /Users/kelvin/Documents -jni kelvin.Java.dynamicso.Hello

4.编写本地方法

在生成的Hello.h头文件中,有需要实现的本地方法名,在实现时,要记得指定参数名称

#include <stdio.h>
#include "Hello.h"

JNIEXPORT jstring JNICALL Java_Hello_sayHello(JNIEnv *env, jclass jc, jstring name)
{
    const char *buf;

    buf = (*env)->GetStringUTFChars(env, name, NULL);
    if (NULL == buf)
    {
        return NULL;
    }

    printf("%s\n", buf);

    (*env)->ReleaseStringUTFChars(env, name, buf);

    return (*env)->NewStringUTF(env, "hello");
}

5.制作动态库

由于是Linux平台,需要制作后缀是.so的动态库,其中,需要指定jni.h的路径,必要时还需要jni_md.h的路径,该文件在jdk的目录中

# gcc -c -fPIC -I/Library/Java/JavaVirtualMachines/jdk1.8.0_91.jdk/Contents/Home/include  Hello.c -o Hello.o
# gcc -shared Hello.o -o libhello.so

6.调用动态库

加载动态库有2种方式:

1)load():需要指定库的绝对路径

2)loadLibrary():需要指定库的相对路径,即java.lib.path

现在jni调用的一切都准备好了,进行最后的调用,有正常的打印输出,表明jni正常调用了

# java Hello
kelvin
hello


以上就是Linux平台的jni调用方式,下一篇介绍Windows平台的jni调用方式。。。

参考资料

JNI之String类型

Jni编程(三)c/c++ 获取java字符串,以及java 获取c/c++创建的对象

一天掌握Android JNI本地编程 快速入门

原文地址:https://www.cnblogs.com/fishbay/p/10016455.html

时间: 2024-10-10 08:52:30

JNI编程实现(Linux)的相关文章

【转】Android JNI编程—JNI基础

原文网址:http://www.jianshu.com/p/aba734d5b5cd 最近看到了很多关于热补的开源项目——Depoxed(阿里).AnFix(阿里).DynamicAPK(携程)等,它们都用到了JNI编程,并且JNI编程也贯穿了Android系统,学会JNI编程对于我们学习研究Android源码.Android安全以及Android安全加固等都是有所帮助的.但是对于我们这些写Android应用的,大部分时间都是在使用Java编程,很少使用C/C++编程,对于JNI编程也了解的比较

android开发教程(4)— jni编程之采用 javah 从java调用C++

用Java调用C/C++代码 当无法用 Java 语言编写整个应用程序时,JNI 允许您使用本机代码.在下列典型情况下,您可能决定使用本机代码: 希望用更低级.更快的编程语言去实现对时间有严格要求的代码. 希望从 Java 程序访问旧代码或代码库. 需要标准 Java 类库中不支持的依赖于平台的特性. 须知:SWIG和javah的区别(强烈推荐) 我看了网上的关于 jni编程 的教程很多,但不尽相同,刚开始会犯迷糊.我想笔者往往忽略了一个关键点,那就是采用了什么方式决定了步骤的流程.有两种生成

android windows 上JNI编程

昨天学习windows上的JNI编程,JNI说白了就是java和c语言的一个互相沟通的桥梁,java可以调用JNI来完成调用C语言实现的方法.JNI的全称是(Java native interface),其实在编程重你只需要将与java交互的函数写出来,其他的C语言内部调用的就可以直接使用C语言相关语法了.闲话少说,开始正题吧. 要想在windroid或者是linux上使用JNI必须要下载NDK的并且指定路径,在windows我们还需要安装一个sygwin,这里我就不再说怎样安装cygwin了,

Android JNI编程—JNI基础

什么是JNI,怎么使用 JNI--Java Native Interface,它是Java平台的一个特性(并不是Android系统特有的).其实主要是定义了一些JNI函数,让开发者可以通过调用这些函数实现Java代码调用C/C++的代码,C/C++的代码也可以调用Java的代码,这样就可以发挥各个语言的特点了.那么怎么使用JNI呢,一般情况下我们首先是将写好的C/C++代码编译成对应平台的动态库(windows一般是dll文件,linux一般是so文件等),这里我们是针对Android平台,所以

编程实现Linux下的ls -l

头文件 #ifndef __FUNC_H__ #define __FUNC_H__ #include<stdio.h> #include <stdio.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <sys/types.h> #include <dirent.h> #include <sys/stat.h> #inc

Android Jni层 创建 linux socket 出错问题解决

问题: 想在Jni层创建 udp socket 与服务端通信,但是没有成功,最后发现竟然是创建socket失败(代码如下) // create socket g_sd = socket(AF_INET, SOCK_DGRAM, 0); if (-1 == g_sd) { perror("socket()"); goto err_socket; } 解决办法: 在 AndroidManifest.xml 文件中,添加访问网络的权限: <uses-permission android

Android jni 编程入门

本文将介绍如何使用eclipse和ndk-build来编写一个基于Android4.4版本的包含有.so动态库的安卓程序. 前提是已经安装和配置好了诸如SDK,NDK等编译环境.下面开始编程! 1 程序逻辑 我们要编写的程序包含两部分:java部分--负责界面和调用JNI native函数:JNI native 部分--负责native函数的具体实现(本文使用C语言). native 函数伪代码如下: ? 1 2 3 4 5 6 7 8 /* funtion: 传入两个整形变量,计算他们之和 r

编程解决Linux下解压zip乱码问题

JDK7 的ZipInputStream新添了一个构造方法,第二个参数可以指定字符集.这样一来我们就能用这个类写一个解压程序解决zip乱码问题了. 下面是代码: package cn.fh; import java.io.File; import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.IOException; import java.nio.charset.Charset; import

编程实现LINUX下目录的层层遍历

/************************************************************************* > File Name: treedir.c > Author: KrisChou > Mail:[email protected] > Created Time: Tue 19 Aug 2014 05:04:50 PM CST *****************************************************