项目中很多代码采用C++编写,配置界面则采用BS结构,使用Java语言进行设置。因此需要实现Java调用C++编写的函数库(dll文件或so文件),采用的技术为JNI(Java Native Interface),对于常用的调用方式在《The Java Native Interface Programmer's Guide and Specification》一书中有详细的描述,不在进行描述。本文中主要介绍在动态链接库中如果含有回调函数作为函数参数的C++函数如何使用JNI实现调用。在C++中函数定义格式为:
typedef int (*UserProcess)(int nEncrptType);
CALLBACK_API int Process(UserProcess cb,int nCompanID);
即如何采用Java语言实现Process函数?下面将分别针对C++和Java语言使用回调函数进行说明。
1.回调函数库
本文中定义了一个简单的回调函数接口,目的是根据公司ID进行加密,加密函数为回调函数。头文件定义如下
#ifdef CALLBACK_EXPORTS
#define CALLBACK_API __declspec(dllexport)
#else
#define CALLBACK_API __declspec(dllimport)
#endif
#ifdef __cplusplus
extern "C"
{
#endif
typedef int (*UserProcess)(int nEncrptType);
CALLBACK_API int Process(UserProcess cb,int nCompanID);
#ifdef __cplusplus
}
#endif
|
其实现文件为
#include "stdafx.h"
#include "CallBack.h"
CALLBACK_API int Process(UserProcess cb,int nCompanID)
{
if(cb == NULL)
{
return -1;
}
cb(nCompanID);
return 0;
}
|
2.C++语言使用回调函数库
用法比较简单,直接实现一个满足UserProcess定义的函数,然后将其传递给Process函数即可,见下文代码内容。
#include "stdafx.h"
#include "../CallBack/CallBack.h"
int WanporProcess(int i)
{
printf("Process = %d\n",i);
return i;
}
int _tmain(int argc, _TCHAR* argv[])
{
Process(WanporProcess,100);
return 0;
}
|
加密函数将公司ID直接输出,未进行其他处理。
3.Java语言使用回调函数
这个过程相对于C++显得比较繁琐,主要包含以下一个过程:
3.1编写回调函数的抽象接口,并实现一个回调类;
3.2在JNI中安装回调接口类
3.3实现C++的回调函数,在此回调函数中获取抽象接口类对象、回调函数的接口名称、调用回调函数
3.4调用process进行回调函数执行
2.5卸载回调类
下面依次实现上述中的内容
3.1回调函数接口定义
package cn.com.wanpor;
//编写回调函数接口
public interface CBInterface{
public abstract int UserProcess(int encryptType);
}
|
3.2实现回调类
//实现回调类
class WanporCB implements CBInterface {
public int UserProcess(int encryptType){
System.out.println("Java = " + 3*encryptType);
return 0;
}
}
|
3.3C++ JNI代码中安装回调类
#include "stdafx.h"
#include "../CallBack/CallBack.h"
#include "cn_com_wanpor_CallBackJNI.h"
typedef struct CBData{
UserProcess m_pfnUserProcess;
jobject m_objInterface;
JNIEnv* m_pEnv;
jobject m_objCallBack;
}CBData;
CBData g_cbData;
JNIEXPORT jint JNICALL Java_cn_com_wanpor_CallBackJNI_setUserCallBack
(JNIEnv * env, jobject obj, jobject cb)
{
g_cbData.m_pEnv = env;
g_cbData.m_objCallBack = env->NewGlobalRef(obj);
g_cbData.m_objInterface = env->NewGlobalRef(cb);
return 0;
}
|
3.4实现C++回调函数
int ProcessJNICB(int nEncrpytType)
{
JavaVM* pVm;
g_cbData.m_pEnv->GetJavaVM(&pVm);
pVm->AttachCurrentThread((void**)&g_cbData.m_pEnv,NULL);
jclass jclsProcess = g_cbData.m_pEnv->GetObjectClass(g_cbData.m_objInterface);
if (jclsProcess == NULL)
{
printf("jclsProcess = NULL\n");
return -1;
}
jmethodID jmidProcess = g_cbData.m_pEnv->GetMethodID(jclsProcess,"UserProcess","(I)I");
if (jmidProcess == NULL)
{
printf("jmidProcess = NULL\n");
return -2;
}
g_cbData.m_pEnv->CallIntMethod(g_cbData.m_objInterface,jmidProcess,nEncrpytType);
pVm->DetachCurrentThread();
return 0;
}
|
3.5执行回调函数
JNIEXPORT jint JNICALL Java_cn_com_wanpor_CallBackJNI_process
(JNIEnv * env, jobject obj, jint nCompanyID)
{
Process(ProcessJNICB,nCompanyID);
return 0;
}
|
3.6卸载回调类
JNIEXPORT jint JNICALL Java_cn_com_wanpor_CallBackJNI_resetUserCallBack
(JNIEnv * env, jobject objCallBack, jobject objInterface)
{
g_cbData.m_pEnv->DeleteGlobalRef(objInterface);
g_cbData.m_pEnv->DeleteGlobalRef(objCallBack);
return 0;
}
|
4.Java测试程序
下面的代码是对JNI接口的测试程序,WanporCB实现了回调接口,回调函数返回值为0,但打印出3*nCompany的值,程序执行函数的顺序是:
创建回调函数接口实例
创建测试对象
安装回调类
执行加密
卸载回调类
代码如下:
package cn.com.wanpor;
//实现回调类
class WanporCB implements CBInterface {
public int UserProcess(int encryptType){
System.out.println("Java = " + 3*encryptType);
return 0;
}
}
public class CallBackJNI{
static{
System.load("E:\\Program Files\\Java\\jdk1.5.0_15\\bin\\CallBackJNI.dll");
}
//设置回调函数
public native int setUserCallBack(CBInterface cb);
//清除回调函数
public native int resetUserCallBack(CBInterface cb);
//执行用户自定义加密
public native int process(int companyID);
//测试程序
public static void main(String[] argv){
CallBackJNI cbj = new CallBackJNI();
WanporCB wcb = new WanporCB();
cbj.setUserCallBack(wcb);
cbj.process(300);
cbj.resetUserCallBack(wcb);
}
}
|
阅读(11232) | 评论(2) | 转发(0) |