JNI备忘API

1 篇文章 0 订阅

最近项目中需要调用别人CVN的加密机,但只提供了C接口,没办法,只能考虑JNI,查了些资料,现在整理如下

首先说说用JNI的弊端:

1、 JNI用于调用native,同时带来的问题是这个项目基本无法移植了。

2、 JNI调用效率比较低,如果数据处理量比较大,建议尽量考虑其他方式。

3、 因为C++等语言比较灵活,特别是有内存管理,而java developer因为JVM的垃圾回收机制,不考虑这些问题,那么调用的时候很可能出现安全问题。

JNI的使用步骤:

1、 新建一个java类TestHello,声明native

	static{
		System.loadLibrary("Test");
	}
	
	public static native void hello(String msg);
声明native方法,并不用去实现,因为他其实就是你要调用DLL里的方法。

2、 用javah命令生成TestHello.h文件

javah -jni com.xixin.TestHello

注意此处,javah在src下执行

注意包名,生成的文件名为:com_xixin_TestHello.h

3、 在VC++6.0中建一个空项目,用于编写com_xixin_TestHello.h内容,以及生成DLL文件

创建Win32 Dynamic-Link Libray即可

简单的DLL工程

工程建好以后加载com_xixin_TestHello.h,并修改

com_xixin_TestHello.h文件的最后加上这段代码

//
JNIEXPORT void JNICALL Java_com_xixin_TestHello_hello(JNIEnv * env, jclass obj, jstring jMsg)
{
    const char *strMsgPtr = env->GetStringUTFChars( jMsg , 0);  
 
       MessageBox( 0, strMsgPtr,"Message box from VC++ ", 0 );
 
       env->ReleaseStringUTFChars( jMsg, strMsgPtr); 
 
}
这段代码其实是调用了win32.dll中的MessageBox方法,这里参数稍作解释:

	(JNIEnv * env, jclass obj, jstring jMsg

jsstring是在JNI中java对应C的string的数据类型

JNIEnv内存比较多,这里暂不详述

obj对象为TestHello对象,如果是static的,则是TestClss.class

最后贴一下com_xixin_TestHello.h文件全貌

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
#include <iostream>
#include <windows.h>
#include <string>

using namespace std;
/* Header for class com_xixin_TestHello */

#ifndef _Included_com_xixin_TestHello
#define _Included_com_xixin_TestHello
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     com_xixin_TestHello
 * Method:    hello
 * Signature: (Ljava/lang/String;)V
 */
JNIEXPORT void JNICALL Java_com_xixin_TestHello_hello
  (JNIEnv *, jclass, jstring);

#ifdef __cplusplus
}
#endif
#endif



//
JNIEXPORT void JNICALL Java_com_xixin_TestHello_hello(JNIEnv * env, jclass obj, jstring jMsg)
{
    const char *strMsgPtr = env->GetStringUTFChars( jMsg , 0);  
 
       MessageBox( 0, strMsgPtr,"Message box from VC++ ", 0 );
 
       env->ReleaseStringUTFChars( jMsg, strMsgPtr); 
 
}
完成以上操作后,BuildAll,在VC++工程的debug目录下可以找到DLL文件,该文件就是我们要用的DLL,此DLL名字应该与前面写的TestHello中的
System.loadLibrary("Test");

"Test"一致

这样就完成了基本操作,然后在java项目main方法中调用TestHello类的hello方法即可


下面提一些可能遇到的问题:

1、Exception in thread "main" java.lang.UnsatisfiedLinkError: no Test in java.library.path

这是由于System.loadLibray("Test");的时候找不到DLL文件,解决办法有一下几种:

a、 配置DLL文件到Path

b、 把DLL文件放到jre的bin目录下

c、 动态设置libray,需要用到如下代码

	public static void addDir(String s) throws IOException {
	    try {
	        Field field = ClassLoader.class.getDeclaredField("usr_paths");
	        field.setAccessible(true);
	        String[] paths = (String[])field.get(null);
	        for (int i = 0; i < paths.length; i++) {
	            if (s.equals(paths[i])) {
	                return;
	            }
	        }
	        String[] tmp = new String[paths.length+1];
	        System.arraycopy(paths,0,tmp,0,paths.length);
	        tmp[paths.length] = s;
	        field.set(null,tmp);
	    } catch (IllegalAccessException e) {
	        throw new IOException("Failed to get permissions to set library path");
	    } catch (NoSuchFieldException e) {
	        throw new IOException("Failed to get field handle to set library path");
	    }
	}

因为libray在java程序运行前只执行一次,所以不能直接设置libray,而是考虑通过反射

2、 中文乱码的问题,查了一些资料,得到一个中文处理的工具类,这段代码是用在com_xixin_TestHello.h中的,代码如下:

char* jstringToWindows( JNIEnv *env, jstring jstr )  
{  
  int length = env->GetStringLength(jstr);  
  const jchar* jcstr = env->GetStringChars(jstr, 0);  
  char* rtn = (char*)malloc(length*2+1);  
  int size = 0;  
  size = WideCharToMultiByte(CP_ACP, 0, (LPCWSTR)jcstr, length, rtn,(length*2+1), NULL, NULL);  
  if( size <= 0 )  
    return NULL;  
  env->ReleaseStringChars(jstr, jcstr);  
  rtn[size] = 0;  
  return rtn;  
}  
jstring WindowsTojstring( JNIEnv *env, const char* str )  
{  
  jstring rtn = 0;  
  int slen = strlen(str);  
  unsigned short* buffer = 0;  
  if( slen == 0 )  
    rtn = env->NewStringUTF(str);   
  else  
  {  
    int length = MultiByteToWideChar(CP_ACP, 0, (LPCSTR)str, slen, NULL, 0);  
    buffer = (unsigned short*)malloc(length*2 + 1);  
    if( MultiByteToWideChar(CP_ACP, 0, (LPCSTR)str, slen, (LPWSTR)buffer, length) >0)  
      rtn = env->NewString((jchar*)buffer, length);  
  }  
  if(buffer)  
  free(buffer);  
  return rtn;  
}  
char* U2G(const char* utf8)
{
  int len = MultiByteToWideChar(CP_UTF8, 0, utf8, -1, NULL, 0);
  wchar_t* wstr = new wchar_t[len+1];
  memset(wstr, 0, len+1);
  MultiByteToWideChar(CP_UTF8, 0, utf8, -1, wstr, len);
  len = WideCharToMultiByte(CP_ACP, 0, wstr, -1, NULL, 0, NULL, NULL);
  char* str = new char[len+1];
  memset(str, 0, len+1);
  WideCharToMultiByte(CP_ACP, 0, wstr, -1, str, len, NULL, NULL);
  if(wstr) delete[] wstr;
  return str;
}
char* G2U(const char* gb2312)
{
  int len = MultiByteToWideChar(CP_ACP, 0, gb2312, -1, NULL, 0);
  wchar_t* wstr = new wchar_t[len+1];
  memset(wstr, 0, len+1);
  MultiByteToWideChar(CP_ACP, 0, gb2312, -1, wstr, len);
  len = WideCharToMultiByte(CP_UTF8, 0, wstr, -1, NULL, 0, NULL, NULL);
  char* str = new char[len+1];
  memset(str, 0, len+1);
  WideCharToMultiByte(CP_UTF8, 0, wstr, -1, str, len, NULL, NULL);
  if(wstr) delete[] wstr;
  return str;
}

当然,这段程序最好放在你的hello方法之前,因为VC++6.0是按先后顺序编译的。


参考资料

http://www.open-open.com/lib/view/open1344438190827.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值