文档章节

Java JNI开发实践记录

secondriver
 secondriver
发布于 2015/09/17 09:25
字数 999
阅读 26
收藏 0

  

    当使用到JNI的时候,基本可以肯定Java的平台移植性注定减弱,接下来记录一次使用Java JNI开发的经历。


    关于Java JNI的相关资料参见:

    http://docs.oracle.com/javase/8/docs/technotes/guides/jni/spec/intro.html


    下面是使用JNI常见三种场景:

    1.在Java应用中标准Java类库不支持平台相关的特性

    2.已经存在用其它语言写好的类库,希望通过Java JNI类访问

    3.需要使用低级语言(如汇编)来实现时效性要求很高的一小部分代码


    这次使用JNI属于第2中场景,由于加解密库使用C来实现的,而在Java应用中使用到其加密后的密文数据,所以解密部分需要此库。


    在1和3这两种场景下使用JNI做法相对容易一些,通常先定义好本地方法,然后通过javah生成头文件,接着用其它语言(如C)来实现相应的功能,而2中场景则需要做一些简单的适配,因为类库已经存在,而在Java中定义好的本地方法并不能直接对应类库的具体实现,所以得通过调用已存在的类库中的方法来实现本地方法。


    在开始之前有一个坑先看看:

    

    本地编译好的动态库头信息:

[ enc]$ readelf -a libfdsi.so 
ELF Header:
  Magic:   7f 45 4c 46 02 01 01 00 00 00 00 00 00 00 00 00 
  Class:                             ELF64
  Data:                              2's complement, little endian *******
  Version:                           1 (current)
  OS/ABI:                            UNIX - System V
  ABI Version:                       0
  Type:                              DYN (Shared object file)
  Machine:                           Advanced Micro Devices X86-64

    

    提供方静态库信息:

 

ELF Header:
  Magic:   7f 45 4c 46 02 02 01 00 00 00 00 00 00 00 00 00 
  Class:                             ELF64
  Data:                              2's complement, big endian
  Version:                           1 (current)
  OS/ABI:                            UNIX - System V
  ABI Version:                       0
  Type:                              REL (Relocatable file)
  Machine:                           PowerPC64


   通过对比应该很清楚了,数据存储模式不同。这里需要明确的是环境一致性很重要。


   接下来来从头到尾实现一个Java调用C的一个解密方法。


  1.定义Java的本地方法(DataDecryt.java)

    

package com.cto;

public class DataDecrypt{

    native public static String decrypt(String data);

}


  2.通过javah命令生成头文件(dd.h)

./javah -classpath . -jni -o dd.h com.cto.DataDecrypt

 

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

#ifndef _Included_com_cto_DataDecrypt
#define _Included_com_cto_DataDecrypt
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     com_cto_DataDecrypt
 * Method:    decrypt
 * Signature: (Ljava/lang/String;)Ljava/lang/String;
 */
JNIEXPORT jstring JNICALL Java_com_cto_DataDecrypt_decrypt
  (JNIEnv *, jclass, jstring);

#ifdef __cplusplus
}
#endif
#endif


  3.定义使用静态库中的方法的头文件(dec.h)

#ifndef __DEC__
#define _DEC__

#ifdef __cplusplus
extern "C"{
#endif

  int ts_comm_dec(const char* in , int inlen, char* out, int* outlen);

#ifdef __cplusplus
}
#endif
#endif

    ts_comm_dec方法即为已经实现了的解密方法。


 4.创建实现dd.h头文件方法的cto.c文件,cto.c中将调用ts_common_dec方法

 

#include <jni.h>
#include <stdio.h>
#include "dec.h"
#include "dd.h"

//about JNI http://doc.oracle.com/javase/1.5.0/docs/guide/jni/spec/functions.html
JNIEXPORT jstring JNICALL Java_com_cto_DataDecrypt_decrypt
  (JNIEnv *env, jclass jc, jstring data){

  char out_str[48];

  const char *enc_str = (*env)->GetStringUTFChars(env, data, 0);
  const jsize  enc_len = (*env)->GetStringUTFLength(env, data);

  int  out_len = sizeof(out_str);

  ts_comm_dec(enc_str, enc_len, out_str, &out_len);

  jstring plain_text = (*env)->NewStringUTF(env, out_str);

  (*env)->ReleaseStringUTFChars(env, data, enc_str);

  return plain_text;
}


 5.编写测试用例(TestDataDecrypt.java)

   这里加载的类库cto即为libcto.so。关于动态库静态库命名规则可百度之。

package com.cto;
import com.cto.DataDecrypt;

public class TestDataDecrypt{

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

    public static void main(String [] args){
        String plainText= DataDecrypt.decrypt(args[0]);

        System.out.println(plainText);
        System.out.println("解密之后的长度是:"+plainText.length());
    }
}


 6.编译动态库

 

gcc cto.c -shared -fPIC -lstdc++  -I~/soft/jdk1.6.0_45/include -I~/soft/jdk1.6.0_45/include/linux -I~/native/enc  libtsbase.a  -o libcto.so


 7.运行测试


./java -cp . -Djava.library.path=. com.cto.TestDataDecrypt Qt96BsMOKGjZ0oiqqhRqcA==               
13********1
解密之后的长度是:11

  解密后的结果和预期一致。


8.需要注意的事项

  命令:javac java javah是同一版本,有时候可能系统中有多个版本的JDK

  权限:从其它地方复制的文件,需要确认读写执行权限

  其它:即便按照文中方法来,同样会遇到各种各样的问题,需要多多查看和发现

本文出自 “野马红尘” 博客,请务必保留此出处http://aiilive.blog.51cto.com/1925756/1623404

© 著作权归作者所有

secondriver
粉丝 10
博文 229
码字总数 233821
作品 0
广州
程序员
私信 提问
native关键字初识--java调用非java代码的接口

Java基础知识——JNI入门介绍(上) Java™ 本机接口(Java Native Interface,JNI)是一个标准的 Java API,它支持将 Java 代码与使用其他编程语言编写的代码相集成。如果您希望利用已有的代...

成长中的菜鸟
2015/02/10
282
0
JNI和NDK的区别

NDK(Native Development Kit)“原生”也就是二进制 android常用的开发方式是java封装的库,而这些库的底层实现是由C/C++实现,如媒体,图形库等 java调用这样实现就需要用JNI(Java Native...

长平狐
2013/01/06
126
0
使用 Java Native Interface 的最佳实践

Java™ 本机接口(Java Native Interface,JNI)是一个标准的 Java API,它支持将 Java 代码与使用其他编程语言编写的代码相集成。如果您希望利用已有的代码资源,那么可以使用 JNI 作为您工...

红薯
2009/07/28
879
0
Android JNI(一)——NDK与JNI基础

本系列文章如下: Android JNI(一)——NDK与JNI基础 Android JNI学习(二)——实战JNI之“hello world” Android JNI学习(三)——Java与Native相互调用 Android JNI学习(四)——JNI的常用方法...

隔壁老李头
2018/05/09
0
0
JNA, Java Native开发利器

下面的文章由转载而来,java与c++的调用基本思路是这样子,但是结合我的使用经验,在使用jna和jnative的时候需要考虑到jdk的版本,jnative跟jdk1.4的结合可能会有点问题,因为我之前选的是j...

彭苏云
2013/04/16
1K
4

没有更多内容

加载失败,请刷新页面

加载更多

OSChina 周六乱弹 —— 如果是个帅小伙你愿意和他出去吗

Osc乱弹歌单(2019)请戳(这里) 【今日歌曲】 小小编辑推荐:《Ghost 》游戏《死亡搁浅》原声 《Ghost 》游戏(《死亡搁浅》原声) - Au/Ra / Alan Walker 手机党少年们想听歌,请使劲儿戳...

小小编辑
30分钟前
28
3
java通过ServerSocket与Socket实现通信

首先说一下ServerSocket与Socket. 1.ServerSocket ServerSocket是用来监听客户端Socket连接的类,如果没有连接会一直处于等待状态. ServetSocket有三个构造方法: (1) ServerSocket(int port);...

Blueeeeeee
今天
6
0
用 Sphinx 搭建博客时,如何自定义插件?

之前有不少同学看过我的个人博客(http://python-online.cn),也根据我写的教程完成了自己个人站点的搭建。 点此:使用 Python 30分钟 教你快速搭建一个博客 为防有的同学不清楚 Sphinx ,这...

王炳明
昨天
5
0
黑客之道-40本书籍助你快速入门黑客技术免费下载

场景 黑客是一个中文词语,皆源自英文hacker,随着灰鸽子的出现,灰鸽子成为了很多假借黑客名义控制他人电脑的黑客技术,于是出现了“骇客”与"黑客"分家。2012年电影频道节目中心出品的电影...

badaoliumang
昨天
16
0
很遗憾,没有一篇文章能讲清楚线程的生命周期!

(手机横屏看源码更方便) 注:java源码分析部分如无特殊说明均基于 java8 版本。 简介 大家都知道线程是有生命周期,但是彤哥可以认真负责地告诉你网上几乎没有一篇文章讲得是完全正确的。 ...

彤哥读源码
昨天
19
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部