`

linux下通过JNI用C/C++中调用JAVA类

 
阅读更多

     最近工作中完成了一个linux平台下利用jmx监控进程的资源状态,并在C++程序中调用的功能。现将C++利用JNI调用java方法的过程总结如下:

实践后总结如下:

 

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)

。。。。。

 

------The End-----

分享到:
评论

相关推荐

    在windows中jni(生成dll)Linux中jni(生成so),java调用曾c++

    该软件主要是介绍,java如何调用才c++程序,并且如何在visaul c++中用命令行cl生成dll文件。

    android 使用jni调用c++

    android配置ndk,使用swig编译生成jni Java接口调用c++方法。这是成功的例子,前提是你要配置一下你的环境变量。如有疑问,可以去查看我的博客。有相关内容。

    JNI(java-c).rar_C/C++层回调Java_JAVA LINUX so_NDK SO_java jni linux

    实现了java-c的互调 1.hello-jin-c是linux下的c实现文件。 (使用NDK编译成.so文件,供java层调用)。...(里面通过JNI调用下面.so中提供的c的实现。) 同理.so里面也可以调用回调java里面实现的文件。

    Java通过JNI调用C++的DLL文件.docx

    Java通过JNI调用C++接口,Demo文档描述(里面有代码),里面有Windows下调用版和linux下调用版,经本人亲自测试,可用。

    springboot+jna/jni调用动态so/dll库

    springboot+jna/jni调用动态so/dll库

    JAVA调用C++的dell应用例子

    Jni(Java Native Interface)是sun提供的java与系统中的原生方法交互的技术(在windows\linux系统中,实现java与native method互调)。目前只能由c/c++实现。后两个都是sourceforge上的开源项目,同时也都是基于jni...

    尚硅谷Android高级开发技术之JNI和NDK开发

    而本套教程正是在Android开发中的核心重点开发技术(JNI/NDK),我们知道,Android的底层是Linux且Java的性能并不如C/C++好,所以在开发一些需要超高性能的App的时候就需要通过JNI来直接调用Android底层框架所提供的...

    如何利用JNI实现Java插件开发

    本文主要工作在于利用JNI实现对已经完成的视频镜头提取C++程序的调用。  整个过程如下所示:  (1)将视频特征提取算法建立相应的windows下的Dll和Linux下的So(这个两个文件大家都知是什么了吧)  (2)利用jni...

    java调用C++编写的动态库dll

    此附件包含了java代码以及vs2013编写的动态库工程。

    java调用c++生成so库

    linux 环境下java调c++生成的so库,对应博客地址: http://blog.csdn.net/woniu211111/article/details/78041868

    java调用dll或so库示例

    完整演示使用JNI开发JAVA和C++交互的例子,本示例提供了在Windows和linux上的编译脚本。请参考JNI的相关文章,如有疑问可留言

    jni 头文件

    c++生成Java可调用的动态库使用的jni ,内含 Windows和Linux两个版本的jni.h,jni_md.h

    详解AndroidStudio JNI +Gradle3.0以上JNI爬坑之旅

    他定义了一些JNI函数,来让开发者可以通过调用这些函数来实现java代码调用C/C++代码。 2.如何使用JNI呢? 我们先将写好的C/C++代码编译成对应平台的动态库(windows是.dll文件,linux是.so文件)。 下面我们来举个...

    java开源包1

    php调用java类 Java批量作业执行框架 MyBatchFramework MyBatchFramework 是一个开源的轻量级的用以创建可靠的易管理的批量作业的Java包,主要特点是多线程、调度、JMX管理和批量执行报表,执行历史等。 SIP协议包...

    JAVA上百实例源码以及开源项目源代码

    通过本源码可以了解到Java如何产生单钥加密的密钥(myKey)、产生双钥的密钥对(keyPair)、如何保存公钥的字节数组、保存私钥到文件privateKey.dat、如何用Java对象序列化保存私钥,通常应对私钥加密后再保存、如何从...

    JAVA上百实例源码以及开源项目

    通过本源码可以了解到Java如何产生单钥加密的密钥(myKey)、产生双钥的密钥对(keyPair)、如何保存公钥的字节数组、保存私钥到文件privateKey.dat、如何用Java对象序列化保存私钥,通常应对私钥加密后再保存、如何从...

    JNI的基础概念

    Java 程序可以通过 JNI(Java Native Interface,Java 本地调用)访问本地的动态链接库,从而扩展其功能、保护关键代码、提高运行效率。可以通过NDK工具生成so库 1. 什么情况下需要使用 JNI Java 本机接口(Java ...

    JNI技术在网络交互中的应用

    Android是一款运行在开放的Linux内核之上的操作系统,其上层应用程序采用Java开发语言,底层核心类库采用C/C++语言编写,开发者能够非常便捷的使用Android SDK调用丰富的API,实现各种功能。不仅如此,Android还支持...

    java开源包8

    php调用java类 Java批量作业执行框架 MyBatchFramework MyBatchFramework 是一个开源的轻量级的用以创建可靠的易管理的批量作业的Java包,主要特点是多线程、调度、JMX管理和批量执行报表,执行历史等。 SIP协议包...

    java开源包4

    php调用java类 Java批量作业执行框架 MyBatchFramework MyBatchFramework 是一个开源的轻量级的用以创建可靠的易管理的批量作业的Java包,主要特点是多线程、调度、JMX管理和批量执行报表,执行历史等。 SIP协议包...

Global site tag (gtag.js) - Google Analytics