C++调用Java语言;
这里的目的主要是了解JavaVM的创建应用,java底层技术,和C++调用Java的基本知识点;
Java调用C/C++一般通过JNI技术实现,这里具体看我的别的文章;
一:环境配置:
win7 64 + VC2010 32 + jdk1.7.0_80 64;
环境配置方法1; // 动态方法;
1:VC2010新建控制台程序或者MFC程序,通过配置管理器,新建64位编译选项;(如果是32位JDK就不需要这个设置了);
2:包含JDK头文件的两个路径到附加包含目录:如,E:\Java\jdk1.7.0_80\include;E:\Java\jdk1.7.0_80\include\win32; 或者经include目录中的内容复制到自己的工程目录;
3:在程序中通过LoadLibrary方法,获取jvm.dll的JNI_CreateJavaVM函数,然后就可以正常编写代码;
环境配置方法2://静态方法
1:同方法1;
2:同方法2;
3:设置JDK中的JRE中的bin和相关目录到系统环境变量;如:E:\Java\jre7\bin\server目录;
或者,在MFC中将E:\Java\jre7\bin\server目录设置为工作目录;
4:通过lib加载,自动加载jvm.dll;
注意:一定有设置系统的环境变量,如果仅仅将jvm.dll复制到自己的工程目录下,虽然编译启动没有问题,但是运行到JNI_CreateJavaVM会崩溃;
建议:用方法一进行C++调用java ,这样方法简单,应用灵活;
二:java程序:
将这个程序放到你的目录,我直接放在D盘;
public class HelloWoldJava
{
public HelloWoldJava()
{
System.out.println("这是构造函数 HelloWoldJava");
}
public static int add( int x, int y)
{
return x+y;
}
public int init( int n )
{
System.out.println("init, 是HelloWoldJava类中的普通成员函数;");
return n;
}
public static void main( String[] args )
{
HelloWoldJava pmainJava = new HelloWoldJava();
pmainJava.init(123);
System.out.println( "Hello Wold java, I test java pac" );
}
}
先javac编译,然后java查看运行结果是否正确;
三:编写C++调用上述代码:
#include <iostream>
#include <jni.h>
using namespace std;
int CppLoadJava()
{
typedef jint(JNICALL *pCreateJavaVM)(JavaVM **, void**, void *);
HINSTANCE hInstance = LoadLibrary("E:\\Java\\jdk1.7.0_80\\jre\\bin\\server\\jvm.dll");
if (hInstance == NULL)
{
return -1;
}
pCreateJavaVM CreateJavaVM = (pCreateJavaVM)GetProcAddress(hInstance, "JNI_CreateJavaVM");
const int nOptions = 4;
JNIEnv *env = NULL;
JavaVM *jvm = NULL;
JavaVMInitArgs vm_args = {0};
JavaVMOption options[nOptions]={0};
long status = 0;
jclass cls = NULL;;
jmethodID mid = NULL;
jint square = NULL;
jboolean not = NULL;
jobject jobj = NULL;
options[0].optionString = "-Djava.compiler=NONE"; /* disable JIT */
options[1].optionString = "-Djava.class.path=.;%JAVA_HOME%/lib;%JAVA_HOME%/lib/tools.jar;D:/";//"-Djava.class.path=c:\myclasses"; /* user classes */ //一定要将刚才编译的类的路径添加;
options[2].optionString = "-Djava.library.path=E:/Java/jdk1.7.0_80/lib"; /* set native library path */
options[3].optionString = "-verbose:jni"; /* print JNI-related messages */
vm_args.version = JNI_VERSION_1_6;
vm_args.options = options;
vm_args.nOptions = nOptions;
vm_args.ignoreUnrecognized = TRUE;
/* Note that in the JDK, there is no longer any need to call
* JNI_GetDefaultJavaVMInitArgs.
*/
status = /*JNI_CreateJavaVM*/CreateJavaVM(&jvm, (void**)&env, &vm_args);
if (status != JNI_ERR)
{
//找到类;(不是类文件)//在java.class.path参数中知道你要获取的类的dir;
cls = env->FindClass("HelloWoldJava");
if ( cls == NULL )
{
return -1;
}
//创建类对象; 相当于在Java中 new HelloWoldJava();
jobject obj_HelloWoldJava = env->AllocObject(cls);
if ( obj_HelloWoldJava == NULL )
{
return -1;
}
{
{
//获取类中静态函数,调用;
mid = env->GetStaticMethodID( cls, "add", "(II)I"); //返回两个数的和;// 对于静态函数,可以不创建对象,前面创建对象只是为了之后的应用方便;
if(mid !=0)
{
square = env->CallStaticIntMethod( cls, mid, 5,5);
std::cout << square << std::endl;
}
}
{
//创建类对象;
//获取类中普通成员函数;
//通过类对象调用普通成员函数;
mid = env->GetMethodID(cls,"init","(I)I"); //输入多少,返回多少;
if(mid !=0)
{
if (0)//方法一,创建类对象;
{
jobj=env->NewObject(cls,mid);
jobject n = env->CallObjectMethod( jobj, mid, 13579 );
//jint n = env->CallIntMethod( jobj, mid, 13579 );
std::cout << int((jint)n) << std::endl;
}
else//方法二,创建类对象,方法二更常用,风方便,对于代码的可阅读等重要;
{
//方法1:
jobject nObj = env->CallObjectMethod( obj_HelloWoldJava, mid, 13579 );
//jobject还不过是class _jobject {};的指针,所以,指针就无所了;
std::cout << jint(nObj) << std::endl;
//方法2:
jint nJint = env->CallIntMethod( obj_HelloWoldJava, mid, 13579 );
std::cout << nJint << std::endl;
}
}
}
}
jvm->DestroyJavaVM();
return 0;
}
FreeLibrary(hInstance);
return -1;
}
void CtestCppJavaDlg::OnBnClickedButton1()
{
// TODO: 在此添加控件通知处理程序代码
int r = CppLoadJava();
}
这里在转一篇文章,可以参考:
《C++调用JAVA方法详解》
转载自:http://public0821.iteye.com/blog/423941
本文主要参考http://tech.ccidnet.com/art/1081/20050413/237901_1.html 上的文章。
C++调用Java主要用到了SUN公司的JNI技术, JNI是Java Native Interface的 缩写。从Java 1.1开始,Java Native Interface (JNI)标准成为java平台的一部分,它允许Java代码和其他语言写的代码进行交互。相关资料见http://java.sun.com/j2se/1.5.0/docs/guide/jni/spec/jniTOC.html
开发环境安装及配置
1.1 安装JDK
到SUN公司网站可以下载到最新版的JDK。下载下来后开始安装,一路选择默认配置即可,本文档中假定安装的是JDK1.4,安装目录为C:\j2sdk1.4.2_15。
1.2 配置VC6.0
通过Visual C++ 6的菜单Tools→Options打开选项对话框。在Directories标签页下添加JDK的相关目录到Include和目录下。
开发测试用到的JAVA类
2.1 开发JAVA类
在硬盘的任意地方新建一个名叫test的文件夹,本文档示例中将test文件夹建立在C盘根目录,然后在里面新建一个名称叫Demo.java的JAVA文件,将下面测试用的代码粘贴到该文件中。
2.2 编译JAVA类
运行CMD控制台程序进入命令行模式,输入命令javac -classpath c:\ c:\test\Demo.java,-classpath参数指定classpath的路径,这里就是test目录所在的路径。(注意:如果你没有将JDK的环境变量设置好,就需要先进入JDK的bin目录下,如下图所示。)
2.3 查看方法的签名
我们知道Java中允许方法的多态,仅仅是通过方法名并没有办法定位到一个具体的方法,因此需要一个字符串来唯一表示一个方法。但是怎么利用一个字 符串来表示方法的具体定义呢?JDK中已经准备好一个反编译工具javap,通过这个工具就可以得到类中每个属性、方法的签名。在CMD下运行javap -s -p -classpath c:\ test.Demo即可看到属性和方法的签名。如下图红色矩形框起来的字符串为方法String append(String str, int i)的签名。
在VC中调用JAVA类
3.1 快速调用JAVA中的函
在VC中新建一个控制台程序,然后新建一个CPP文件,将下面的代码添加到该文件中。运行该文件,即可得到Demo类中String append(String str, int i)函数返回的字符串。
3.2 调用步骤分析及注意事项
a、加载jvm.dll动态库,然后获取里面的JNI_CreateJavaVM函数。这个步骤也可以通过在VC工程的LINK标签页里添加对jvm.lib的连接,然后在环境变量里把jvm.dll所在的路径加上去来实现。但后面这种方法在部署的时候会比前一个方法麻烦。
b、利用构造好的参数,调用JNI_CreateJavaVM函数创建JVM。JNI_CreateJavaVM函数内部会自动根据jvm.dll的路径来获取JRE的环境,所以千万不要把jvm.dll文件拷贝到别的地方,然后再通过LoadLibrary函数导入。
c、JVM创建成功后,JNI_CreateJavaVM函数会传出一个JNI上下文环境对象(JNIEnv),利用该对象的相关函数就可以调用JAVA类的属性和方法了。
d、以上面的代码为例:先调用JNIEnv的FindClass方法,该函数传入一个参数,该参数就是java类的全局带包名的名称,如上面示例中的test/Demo表示test包中的Demo类。这个方法会在你创建JVM时设置的classpath路径下找相应的类,找到后就会返回该类的class对象。 Class是JAVA中的一个类,每个JAVA类都有唯一的一个静态的Class对象,Class对象包含类的相关信息。为了使FindClass方法能找到你的类,请确保创建JVM时-Djava.class.path=参数设置正确。注意:系统环境变量中的CLASSPATH对这里创建JVM没有影响,所以不要以为系统CLASSPATH设置好了相关路径后这里就不用设置了。
e、利用FindClass返回的class对象,调用GetMethodID函数可以获得里面方法的ID,在这里GetMethodID函数传入了三个参数:第一个参数是class对象,因为方法属于某个具体的类;第二个参数是方法的名称;第三个参数是方法的签名,这个签名可以在前面3.3中介绍的方法获得。
f、利用class对象,可以通过调用AllocObject函数获得该class对象对应类的一个实例,即Demo类的对象。
g、利用上面获取的函数ID和Demo类的对象,就可以通过CallObjectMethod函数调用相应的方法,该函数的参数跟printf函数的参数一样,个数是不定的。第一个参数是类的对象;第二个参数是要调用的方法的ID;后面的参数就是需要传给调用的JAVA类方法的参数,如果调用的JAVA类方法没有参数,则调用CallObjectMethod时传前两个参数就可以了。
h、从上面的示例中可以看到,在调用JAVA的方法前,构造传入的字符串时,用到了NewJString函数;在调用该方法后,对传出的字符串调用了JstringToCString函数。这是由于Java中所有的字符都是Unicode编码,但是在本地方法中,例如用VC编写的程序,如果没有特殊的定义一般都没有使用Unicode的编码方式。为了让本地方法能够访问Java中定义的中文字符及Java访问本地方法产生的中文字符串,定义了两个方法用来做相互转换。
i、避免在被调用的JAVA类中使用静态final成员变量,因为在C++中生成一个JAVA类的对象时,静态final成员变量不会像JAVA中new对象时那样先赋值。如果出现这种情况,在C++中调用该对象的方法时会发现该对象的静态final成员变量值全为0或者null(根据成员变量的类型而定)。
3.3 调用JAVA中的静态方法
3.4 调用JAVA中的静态属性
3.5 调用JAVA中的带参数构造函数
3.6 传入传出数组
3.7 异常处理
由于调用了Java的方法,因此难免产生操作的异常信息,如JAVA函数返回的异常,或者调用JNI方法(如GetMethodID)时抛出的异常。这些异常没有办法通过C++本身的异常处理机制来捕捉到,但JNI可以通过一些函数来获取Java中抛出的异常信息。
多线程
4.1 多线程中注意事项
JNIEnv和jobject对象都不能跨线程使用
对于jobject,解决办法是
a、m_obj = m_env->NewGlobalRef(obj);//创建一个全局变量
b、jobject obj = m_env->AllocObject(m_cls);//在每个线程中都生成一个对象
对于JNIEnv,解决办法是在每个线程中都重新生成一个env
JNIEnv *env;
m_jvm->AttachCurrentThread((void **)&env, NULL);