原文:http://www.diybl.com/course/3_program/c++/cppjs/20110103/552247.html
项目中很多代码采用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
",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
");
- return -1;
- }
- jmethodID jmidProcess = g_cbData.m_pEnv->GetMethodID(jclsProcess,"UserProcess","(I)I");
- if (jmidProcess == NULL)
- {
- printf("jmidProcess = NULL
");
- 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);
- }
-
- }