摘要:

1 java类生成c头文件和库文件

2 对于c/c++程序,启动时先启动jvm,然后获得对应的java类的对象和方法。然后正常使用。

最近正在做一个C/C++调用java的程序,这里说的调用java不是使用方式 exec(/path/to/java,.....),而是调用一个class文件中的一个特定的函数。

实践后总结如下:

1. 安装 jdk

2. 安装gcc(linux自带有的就无需安装了)

利用JNI(java native interface),来实现动态建立java runtime environment.

第一,C/C++程序中包含头文件"jni.h"

#include <jni.h> 一般在JAVA_HOME/include 目录下。

调用jni.h中的方法建立runtime env 然后调用java 程序。

第二,编译

g++ -o testjava testjava.cpp -I${JAVA_HOME}/include -I${JAVA_HOME}/include/linux -L${JRE_HOME}/lib/i386/client -ljvm

以上就是大致思路,现详细说明过程如下:

#####################################################################################

一、安装配置Java环境

我的linux是RedHat Enterprise linux 5, 内核版本2.6.18

在Linux系统中安装Java比较简单。可以访问Java download网站或自由软件库等,选择你所有安装的操作系统类型(Linux,Linux AMD64,Solaris等)。一旦你已经选择下载文件──要么是自解压缩执行文件,要么是自解压缩的RPM文件,你都可以安装它。我下载的是jdk-1_5_0_06-linux-i586.bin:

# mkdir /usr/local/java

# cd /usr/local/java

# cp /home/soft/jdk-1_5_0_06-linux-i586.bin ./

# chmod u+x jdk-1_5_0_06-linux-i586.bin

# ./jdk-1_5_0_06-linux-i586.bin

运行完后生成jdk1.5.0_06目录,jdk被安装在/usr/local/java/jdk1.5.0_06/。运行以下执行代码将得到一个测试结果:

# cd jdk1.5.0_06/bin

[root@localhost bin]# ./java -version

java version "1.5.0_06"

Java(TM) 2 Runtime Environment, Standard Edition (build 1.5.0_06-b05)

Java HotSpot(TM) Client VM (build 1.5.0_06-b05, mixed mode, sharing)

  

为了能够使用Java,需要设置如下环境变量:

JAVA_HOME=/usr/local/java/jdk1.5.0_06

PATH=$PATH:/usr/local/java/jre1.5.0_05/bin

export JAVA_HOME PATH

export JRE_HOME=/usr/lib/jvm/java-1.4.2-gcj-1.4.2.0/jre

export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:$JRE_HOME/lib/i386:$JRE_HOME/lib/i386/client

注意JRE_HOME的配置,若机器上没有jre环境,则安装jre,安装方法类似安装jdk

设置完后可以查看变量的值

[root@localhost bin]# echo $JAVA_HOME

/usr/local/java/jdk1.5.0_06

[root@localhost bin]# echo $PATH

/usr/kerberos/sbin:/usr/kerberos/bin:/usr/local/bin:/bin:/usr/bin:/home/zhangp/bin:/usr/lib/jvm/java-1.4.2-gcj-1.4.2.0/jre/bin:/usr/local/java/jdk1.5.0_06/bin

二、编写简单的Java程序

package com.test;

public class MyTest {

 public MyTest(){

  super();

 }

public static int add(int a,int b) {

  return a+b;

 }

public boolean judge(boolean bool) {

   return !bool;

 }

}

编译Java程序:

#javac MyTest.java

编译之后生成MyTest.class,将其放置于当前目录的com/test目录下,C++程序的JNI调用时会使用相关方法在com/test目录下查找该class。

三、C++程序

#include <stdio.h>

#include <iostream>

#include <jni.h>

#include <stdlib.h>

#include <assert.h>

jstring stoJstring(JNIEnv* env, const char* pat)

{

  jclass strClass = env->FindClass("java/lang/String");

  jmethodID ctorID = env->GetMethodID(strClass, "<init>", "([BLjava/lang/String;)V");

  jbyteArray bytes = env->NewByteArray(strlen(pat));

  env->SetByteArrayRegion(bytes, 0, strlen(pat), (jbyte*)pat);

  jstring encoding = env->NewStringUTF("utf-8");

  return (jstring)env->NewObject(strClass, ctorID, bytes, encoding);

}

char* jstringTostring(JNIEnv* env, jstring jstr)

{

 char* rtn = NULL;

 jclass clsstring = env->FindClass("java/lang/String");

 jstring strencode = env->NewStringUTF("utf-8");

 jmethodID mid = env->GetMethodID(clsstring, "getBytes", "(Ljava/lang/String;)[B");

 jbyteArray barr= (jbyteArray)env->CallObjectMethod(jstr, mid, strencode);

 jsize alen = env->GetArrayLength(barr);

 jbyte* ba = env->GetByteArrayElements(barr,JNI_FALSE);

 if(alen > 0){

  rtn = (char*)malloc(alen + 1);

  memcpy(rtn, ba, alen);

  rtn[alen] = 0;

 }

 env->ReleaseByteArrayElements(barr, ba, 0);

 return rtn;

}

using namespace std;

int main()

{

 JavaVMOption options[2];

 JNIEnv *env;

 JavaVM *jvm;

 JavaVMInitArgs vm_args;

 long status;

 jclass cls;

 jmethodID mid;

 jint square;

 jboolean jnot;

 jobject jobj;

options[0].optionString = "-Djava.compiler=NONE";

 options[1].optionString = "-Djava.class.path=.";

 //options[2].optionString = "-verbose:jni"; //用于跟踪运行时的信息

vm_args.version = JNI_VERSION_1_4; // JDK版本号

 vm_args.nOptions = 2;

 vm_args.options = options;

 vm_args.ignoreUnrecognized = JNI_TRUE;

status = JNI_CreateJavaVM(&jvm, (void**)&env, &vm_args);

if(status != JNI_ERR){

  printf("create java jvm success\n");

  cls = env->FindClass("com/test/MyTest");  // 在这里查找ava类

  if(cls !=0){

   printf("find java class success\n");

   // 构造函数

   mid = env->GetMethodID(cls,"<init>","()V");

   if(mid !=0){

    jobj=env->NewObject(cls,mid);

    std::cout << "init ok" << std::endl;

   }

       

   // 调用add函数

   mid = env->GetStaticMethodID( cls, "add", "(II)I");

   if(mid !=0){

    square = env->CallStaticIntMethod( cls, mid, 5,5);

    std::cout << square << std::endl;

   }

// 调用judge函数

   mid = env->GetMethodID( cls, "judge","(Z)Z");

   if(mid !=0){

    jnot = env->CallBooleanMethod(jobj, mid, 1);

    if(!jnot) std::cout << "Boolean ok" << std::endl;

}

  }

  else{

   fprintf(stderr, "FindClass failed\n");

  }

 

  jvm->DestroyJavaVM();

  fprintf(stdout, "Java VM destory.\n");

  return 0;

 }

 else{

  printf("create java jvm fail\n");

  return -1;

 }

}

编译该C++程序(前提:Java环境已设置好,即JAVA_HOME、PATH、JRE_HOME、LD_LIBRARY_PATH)

[root@localhost jni]# g++ -o testjava testjava.cpp -I${JAVA_HOME}/include -I${JAVA_HOME}/include/linux -L${JRE_HOME}/lib/i386/client -ljvm

编译好后可以用ldd testjava查看其使用的链接库的正确性。

运行:

[root@localhost jni]# ./testjava

create java jvm success

find java class success

init ok

10

Boolean ok

Java VM destory.

JRE_HOME和LD_LIBRARY_PATH要设置好,编译C++程序时要使用JRE_HOME下面的libjvm.so动态库(一开始我使用网上说的使用JAVA_HOME目录下的libjvm.so,结果出现下面错误

# An unexpected error has been detected by HotSpot Virtual Machine:

#

#  SIGSEGV (0xb) at pc=0xb6d3dbe3, pid=14454, tid=2773482416

#

# Java VM: Java HotSpot(TM) Server VM (1.5.0_11-b03 mixed mode)

。。。。。

转自:http://blog.sina.com.cn/s/blog_48eef8410100fjxr.html

参考文档:使用 Java Native Interface 的最佳实践

C++调用java的更多相关文章

  1. oracle调用JAVA类的方法

    导入jar包 在oracle中导入需要的jar包,我们把编辑好的java类打成jar包,直接在oarcle里面写简单的调用就可以了,  1.操作系统需要拥有支持loadjava命令的jdk.  2.加 ...

  2. android webView开发之js调用java代码示例

    1.webView设置 webView.getSettings().setJavaScriptEnabled(true);//设置支持js webView.addJavascriptInterface ...

  3. Android 使用js调用Java

    效果如: 主要用到一个接口类:MyObject package com.example.jsdemo; import android.content.Context; import android.s ...

  4. 关于c#调用java中间件api的几个问题

    由于项目需要,做的c#客户端数据库连接串首先肯定不能写死的程序里(数据库很容易被攻击,我们的项目半年改了几次密码...) 放置在配置文件内,都可以看得到,最开始想法将配置文件加密,老师说加密过的文件还 ...

  5. C++调用JAVA方法详解

    C++调用JAVA方法详解          博客分类: 本文主要参考http://tech.ccidnet.com/art/1081/20050413/237901_1.html 上的文章. C++ ...

  6. JNI中C调用Java方法

    背景需求 我们需要在JNI的C代码调用Java代码.实现原理:使用JNI提供的反射借口来反射得到Java方法,进行调用. JNI关键方法讲解. 1. 在同一个类中,调用其他方法 JNIEXPORT v ...

  7. 规则引擎集成接口(七)规则引擎调用Java类

    规则引擎调用Java类 通过myEclipse编写一个简单工程,其中方法是两数相加等到结果,既结果1=输入值1+输入值2.实现规则调用外部接口的方法有三种. 1:接口实例:在myEclipse中制作一 ...

  8. oracle 调用java

    1.首先在PL/SQL中创建JAVA类,并编译 例如:这个是到的一个查询目录下面文件列表的java类 创建此java 类用: create or replace and compile java so ...

  9. C#调用Java方法(详细实例)

    C#可以直接引用C++的DLL和转换JAVA写好的程序.最近由于工作原因接触这方面比较多,根据实际需求,我们通过一个具体例子把一个JAVA方法转换成可以由C#直接调用的DLL C#调用c++ C#调用 ...

  10. JNI系列——C文件中的方法调用Java中方法

    1.创建xxx.jni包并在该包下实现一些Java的方法,和要调用的本地方法 2.实现MainActivity中的按钮点击事件-即点击按钮调用本地的方法 3.在C文件中的方法中回调Java的方法 3. ...

随机推荐

  1. 使用observable数组(Working with observable arrays)

    observable数组(observable arrays) 如果你要探测和响应一个对象的变化,你应该用observables.如果你需要探测和响应一个集合对象的变化,你应该用observableA ...

  2. 解决网页在手机浏览器打开不停刷新的方案(百度的ua自动转向js问题)

    一:发现问题 原有可能是网站内挂了一个百度的ua自动转向js,手机访问的话会被自动转到feiyujd.com,然后又被转到www点feiyujd点com,这样反复死循环.就形成了一直在刷新,网站一闪一 ...

  3. GRPC: set up..

    get the grpc source file.. git clone https://github.com/grpc/grpc git submodule update --init --recu ...

  4. android开发注意点

    一.Android编码规范 1.java代码中不出现中文,最多注释中可以出现中文 2.局部变量命名.静态成员变量命名 只能包含字母,单词首字母出第一个外,都为大写,其他字母都为小写 3.常量命名 只能 ...

  5. SourceTree基础

    克隆(clone):从远程仓库URL加载创建一个与远程仓库一样的本地仓库 提交(commit):将暂存文件上传到本地仓库(我们在Finder中对本地仓库做修改后一般都得先提交一次,再推送) 检出(ch ...

  6. VB.net结束进程

    Public Class Form1 Private Sub Button1_Click(sender As Object, e As EventArgs) Handles Button1.Click ...

  7. hdu_1969_pie(二分)

    题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=1969 题意:看了老半天,就是有N个饼,要分给f+1个人,每个人只能一样多,不能拼凑,多余的丢弃,问每个 ...

  8. C++ 中 const和define的区别

    来源网址:http://wujiangping.blog.163.com/blog/static/195182011201255115125205/ 请区别用#define命令定义的符号常量和用con ...

  9. LeetCode OJ 202. Happy Number

    Write an algorithm to determine if a number is "happy". A happy number is a number defined ...

  10. Bank Interest

    Bank Interest Time Limit : 2000/1000ms (Java/Other)   Memory Limit : 131072/65536K (Java/Other) Tota ...