java调用c++ dll出现中文乱码

时间:2023-02-10 10:22:28

最近的开发用到了使用java调用本机动态连接库的功能,将文件路径通过java调用C++代码对文件进行操作。在调用中如果路径中包含有中文字符就会出现问题,程序运行就会中止。下面用一个小例子,来说明记录下解决的方法。

java中传入一个字符串,调用c++代码将字符串输出

public class CommonUtil
{
	static
	{
		System.loadLibrary("nativeTest");
	}
	public native static void Print(String str);
	public static void main(String args[])
	{
		CommonUtil.Print("中文乱码");
	}
}

执行javac CommonUtil.java和javah CommonUtil两条命令。会生成一个CommonUtil.h的c++头文件。CommonUtil.h的源码如下

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

#ifndef _Included_CommonUtil
#define _Included_CommonUtil
#ifdef __cplusplus
extern "C" {
#endif
/*
 * Class:     CommonUtil
 * Method:    Print
 * Signature: (Ljava/lang/String;)V
 */
JNIEXPORT void JNICALL Java_CommonUtil_Print
  (JNIEnv *, jclass, jstring);

#ifdef __cplusplus
}
#endif
#endif

使用vs2005新建一个c++ dll的工程,将CommonUtil.h加入到项目中,再新建一个.cpp文件,用于实现Java_CommonUtil_Print这个函数,实现代码如下:

#include "CommonUtil.h"
#include <iostream>
using namespace std;

JNIEXPORT void JNICALL Java_CommonUtil_Print
			(JNIEnv *env, jclass obj, jstring jStr)
{
	const char *localStr = env->GetStringUTFChars(jStr,NULL);
	cout<<localStr<<endl;
}
在编译中需要加入java自带的c++头文件,否则比如像JNIEnv这样的类就会找不到,我用的是jdk1.6,所以了"C:\Program Files\Java\jdk1.6.0_10\include;C:\Program Files\Java\jdk1.6.0_10\include\win32"到项目属性中。

java调用c++ dll出现中文乱码

编译后生成一个dll,将dll拷到刚才编译的.class所在的文件夹中(做为一个简单的测试,没有使用包,如果使用包情况会略有不同)。

执行命令java CommonUtil输出如下

java调用c++ dll出现中文乱码

现在还完全搞清楚出现乱码的情况,不过网上有将java的utf编码转换成gb2312的代码。下面是转换的代码,代码来源:http://blog.csdn.net/yiyaaixuexi/article/details/6173592

char* jstringToWindows( JNIEnv *env, jstring jstr )
{ //UTF8/16转换成gb2312
  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;
}
将Java_CommonUtil_Print改成如下:

JNIEXPORT void JNICALL Java_CommonUtil_Print
			(JNIEnv *env, jclass obj, jstring jStr)
{
	char *localStr = jstringToWindows(env, jStr);
	cout<<localStr<<endl;
	free(localStr);
}
重新编译,生成后的dll再拷到.class所在的文件夹中。

执行java CommonUtil

java调用c++ dll出现中文乱码
运行正常