首页 > 代码库 > 【转】c++调用java
【转】c++调用java
转:http://blog.sina.com.cn/s/blog_62b2318d0101h5j1.html
c调用java走的也是jni,具体流程:
1、初始化jvm
2、加载你要调用的java类
3、获取类中的函数
4、调用函数
我们一步步来,首先编写一个java类(没有它什么都是扯淡~~),我写了一个最简单的
package com.cjni;
public class CJniTest
{
public CJniTest() {
super();
}
public static int add(int a, int b) {
return a+b;
}
public boolean ju(boolean b) {
return !b;
}
}
放到com/cjni文件夹下,编译后得到class文件。OK,java类解决了。
接下来就是C代码了:
#include "stdafx.h"
#include "jni.h"
#include "jni_md.h"
#include <iostream>
using namespace std;
#pragma comment(lib, "jvm.lib")
void callFunction(JNIEnv* env)
{
jclass cls;
jmethodID mid;
jboolean b;
jobject jobj;
jint square;
//加载类
cls = env->FindClass("com/cjni/CJniTest");
if(cls != 0) {
//获取静态方法add,第三个参数为sig
mid = env->GetStaticMethodID(cls, "add", "(II)I");
if(mid != 0) {
//调用方法
square = env->CallStaticIntMethod(cls, mid, 10, 20);
cout<<"add:"<<square<<endl;
}
//获取构造函数
mid = env->GetMethodID(cls, "<init>", "()V");
if(mid != 0) {
//创建对象
jobj = env->NewObject(cls, mid);
cout<<"init ok"<<endl;
}
//获取普通方法ju
mid = env->GetMethodID(cls, "ju", "(Z)Z");
if(mid != 0) {
//调用方法
b = env->CallBooleanMethod(jobj, mid, 1);
if(!b)
cout<<"ju not true"<<b<<endl;
}
}
}
void testClassFile()
{
JavaVMOption options[1];
JNIEnv* env;
JavaVM* jvm;
JavaVMInitArgs vm_args;
long status;
//设置class目录为当前文件夹,也就是说将class文件放在当前目录下就行了
options[0].optionString = "-Djava.class.path=.";
vm_args.options = options;
vm_args.nOptions = 1;
vm_args.version = JNI_VERSION_1_6;
//创建JVM
status = JNI_CreateJavaVM(&jvm, (void**)&env, &vm_args);
if(status != JNI_ERR) {
callFunction(env);
//清除JVM
jvm->DestroyJavaVM();
}
else {
cout<<"create java vm error!"<<endl;
}
}
void testJarFile()
{
JavaVMOption options[1];
JNIEnv* env;
JavaVM* jvm;
JavaVMInitArgs vm_args;
long status;
//将jar包加入class路径中
options[0].optionString = "-Djava.class.path=.;cjni.jar";
vm_args.options = options;
vm_args.nOptions = 1;
vm_args.version = JNI_VERSION_1_6;
//创建jvm
status = JNI_CreateJavaVM(&jvm, (void**)&env, &vm_args);
if(status != JNI_ERR) {
callFunction(env);
jvm->DestroyJavaVM();
}
else {
cout<<"create java vm error!"<<endl;
}
}
int _tmain(int argc, _TCHAR* argv[])
{
// testClassFile();
testJarFile();
return 0;
}
使用vs2010编写,之前要设置好头文件,库文件的搜索路径。
程序使用两种方式加载class,一种是直接使用class文件,另一种是将class打成jar包,没什么太大区别。
还有一个要注意的地方是当调用env->GetStaticMethodID或env->GetMethodID的第三个参数为sig,这个值可以通过如下命令查看:
F:\source\vc\CCallJava\com\cjni>javap -p -s CJniTest
Warning: Binary file CJniTest contains com.cjni.CJniTest
Compiled from "CJniTest.java"
public class com.cjni.CJniTest {
public com.cjni.CJniTest();
Signature: ()V
public static int add(int, int);
Signature: (II)I
public boolean ju(boolean);
Signature: (Z)Z
}
上面的Signature就是sig值,至于为什么会需要这个参数,原因是java支持函数重载,所以只是用函数名是无法定位一个函数的,sig的功能就是协助定位一个函数,其命名规则是根据函数参数与返回值构造一个字符串,将其与函数名共同标示一个函数,括号里面的参数类型,括号外的是返回值类型。(话说C++中对于重载函数也是需要处理的,编译后生成的乱七八糟的函数名,调试的时候哦很无语)
以上是jni的基本用法,详细的还有很多,有了这些知识,足够我们在cocos2d-x中调用java了。
创建一个android平台的cocos2d-x项目,我创建了一个例子项目叫cardmap,那么在项目中的cardmap\proj.android\src\org\cocos2dx\cardmap目录是存放java文件的地方,手工写个java类,
package org.cocos2dx.cardmap;
public class CardMapJni
{
public CardMapJni() {
super();
}
public static int add(int a, int b) {
return a+b;
}
public boolean ju(boolean b) {
return !b;
}
}
和上面的很像?只是package和类名不同,嘿嘿
然后打开大名鼎鼎的HelloWorldScene.cpp,
在最上面加入如下代码:
#include "platform/android/jni/JniHelper.h"
#include <jni.h>
#include <android/log.h>
主要是将jni相关头文件包含进来。
然后随便写个函数:
int testJNI()
{
JniMethodInfo method;
bool b = JniHelper::getStaticMethodInfo(method, "org.cocos2dx.cardmap.CardMapJni", "add", "(II)I");
if(b) {
jint square = method.env->CallStaticIntMethod(method.classID, method.methodID, 50, 60);
return square;
}
return 0;
}
代码很简单,调用add方法并返回结果。
对代码做一下简要分析:
JniMethodInfo 是在JniHelper.h中声明的,如下:
typedef struct JniMethodInfo_
{
JNIEnv * env;
jclass classID;
jmethodID methodID;
} JniMethodInfo;
其包装了env,class与method。
从类中获取一个方法使用JniHelper::getStaticMethodInfo或JniHelper::getMethodInfo,函数会将调用函数所需的信息放入第一个参数中,这样就可以通过JniMethodInfo调用对应的函数了。
实际开发中,可以针对每个需要调用的java类,封装与其对应的c++类,在初始化时将JniMethodInfo封装好,在需要的时候直接调用就行了~~