1:本文是接上文JNI(1)之java调用c++,c++的JNI库文件的加入可参考上文,否则不能编译成功的哦,
依然开始java端程序:
package com.myjava;
public class MyDrawView {
private String emptyHour;
/**
* 初始化
*/
public void init(){
emptyHour="申酉哈哈";
}
public static void main(String[] args) {
MyDrawView mv=new MyDrawView();
mv.init();
}
}
2:c++端的程序
#include "windows.h"
#include "jni.h"
#include <string>
#include <iostream>
using namespace std;
jstring NewJString(JNIEnv *env, const char* str);
string JStringToCString (JNIEnv *env, jstring str);
int main()
{
//定义一个函数指针,下面用来指向JVM中的JNI_CreateJavaVM函数
typedef jint (WINAPI *PFunCreateJavaVM)(JavaVM **, void **, void *);
int res;
JavaVMInitArgs vm_args;
JavaVMOption options[3];
JavaVM *jvm;
JNIEnv *env;
/*设置初始化参数*/
options[0].optionString = "-Djava.compiler=NONE";
//设置classpath,设置你java项目包的路径
options[1].optionString = "-Djava.class.path=E:\\daima\\a-java\\bin\\";
//设置显示消息的类型,取值有gc、class和jni,如果一次取多个的话值之间用逗号格开,如-verbose:gc,class
//该参数可以用来观察C++调用JAVA的过程,设置该参数后,程序会在标准输出设备上打印调用的相关信息
options[2].optionString = "-verbose:NONE";//jni
//设置版本号,版本号有JNI_VERSION_1_1,JNI_VERSION_1_2和JNI_VERSION_1_4
//选择一个根你安装的JRE版本最近的版本号即可,不过你的JRE版本一定要等于或者高于指定的版本号
vm_args.version = JNI_VERSION_1_4;
vm_args.nOptions = 3;
vm_args.options = options;
//该参数指定是否忽略非标准的参数,如果填JNI_FLASE,当遇到非标准参数时,JNI_CreateJavaVM会返回JNI_ERR
vm_args.ignoreUnrecognized = JNI_TRUE;
//加载JVM.DLL动态库,如果字符集为Unicode字串常数,加L“...”;例如:wchar_t *wfilename= L“HelloWorld”;否则不加L
HINSTANCE hInstance =::LoadLibrary(L"C:\\Program Files\\Java\\jre1.6.0_04\\bin\\client\\jvm.dll");
if (hInstance == NULL)
{
return false;
}
//取得里面的JNI_CreateJavaVM函数指针(函数检索指定的动态链接库(DLL)中的输出库函数地址)
PFunCreateJavaVM funCreateJavaVM = (PFunCreateJavaVM)::GetProcAddress(hInstance, //DLL模块句柄
"JNI_CreateJavaVM");//函数名
//调用JNI_CreateJavaVM创建虚拟机
res = (*funCreateJavaVM)(&jvm, (void**)&env, &vm_args);
if (res < 0)
{
return -1;
}
jclass hiClass;
jmethodID cid;
//类的路径在之前设置的classpath的路径基础上
//查找class类,返回JAVA类的CLASS对象
hiClass =env->FindClass("com/myjava/MyDrawView");
if(hiClass==NULL){
printf("未找到class");
return NULL;
};
//第2参数,构造方法为<init>
cid =env->GetMethodID(hiClass, "<init>", "()V");
if(cid==NULL)return NULL;
jobject jo=env->NewObject(hiClass,cid);
jmethodID mid =env->GetMethodID(hiClass, "init", "()V");
if (mid == NULL) {
printf("未找到方法init");
return -1;
}
env->CallVoidMethod(jo,mid);
//获取类中的方法,最后一个参数是方法的签名,通过javap -s -p 文件名可以获得
jfieldID fid = env->GetFieldID(hiClass, "emptyHour", "Ljava/lang/String;");
jstring msg = (jstring) env->GetObjectField(jo, fid);
cout<<JStringToCString(env, msg);
env->DeleteLocalRef(hiClass);
//销毁虚拟机并释放动态库
jvm->DestroyJavaVM();
::FreeLibrary(hInstance);
return 0;
}
string JStringToCString (JNIEnv *env, jstring str)
{
if(str==NULL)
{
return "";
}
//在VC中wchar_t是用来存储宽字节字符(UNICODE)的数据类型
int len = env->GetStringLength(str);
wchar_t *w_buffer = new wchar_t[len+1];
char *c_buffer = new char[2*len+1];
ZeroMemory(w_buffer,(len+1)*sizeof(wchar_t));
//使用GetStringChars而不是GetStringUTFChars
const jchar * jcharString = env->GetStringChars(str, 0);
wcscpy(w_buffer,(wchar_t*)jcharString);
env->ReleaseStringChars(str,jcharString);
ZeroMemory(c_buffer,(2*len+1)*sizeof(char));
//调用字符编码转换函数(Win32 API)将UNICODE转为ASCII编码格式字符串
len = WideCharToMultiByte(CP_ACP,0,w_buffer,len,c_buffer,2*len,NULL,NULL);
string cstr = c_buffer;
delete[] w_buffer;
delete[] c_buffer;
return cstr;
}
jstring NewJString(JNIEnv *env, const char *str)
{
if(!env || !str)
{
return 0;
}
int slen = strlen(str);
jchar* buffer = new jchar[slen];
int len = MultiByteToWideChar(CP_ACP,0,str,strlen(str),(LPWSTR)buffer,slen);
if(len>0 && len < slen)
{
buffer[len]=0;
}
jstring js = env->NewString(buffer,len);
delete [] buffer;
return js;
}
若读者看后疑惑较多建议看看JNI基础,本人提供实例参考而已。
至此运行c++项目即可,本人用的是vs2010和eclipse