- 浏览: 332189 次
- 性别:
- 来自: 北京
文章分类
最新评论
-
longge424:
你好,可否将saml与spring security结合使用的 ...
SAML介绍 -
bnmnba:
mark
(转)Ant 学习笔记(一) -
redhacker:
写的还不错!谢谢!
SAML介绍 -
linluxian:
SAML介绍 -
hyj1254:
有效,good.
Linux创建目录与删除目录命令具体分析
最近工作中完成了一个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-----
发表评论
-
一个不错的shell 脚本教程
2015-12-16 17:01 0Linux中有好多中不同的shell,但是通常我们使用bas ... -
Linux下查看文件和文件夹大小
2015-08-27 17:16 765当磁盘大小超过标准时会有报警提示,这时如果掌握df和du命令 ... -
(转)Dubbo与Zookeeper、SpringMVC整合和使用(负载均衡、容错)
2015-06-04 16:49 1849互联网的发展,网站应用的规模不断扩大,常规的垂直应用架构已 ... -
AWK处理两个文件公共字段
2015-06-03 16:36 4193在文件score.txt中存有如下数据:(姓名 分数) ... -
maven+springMVC+mybatis+junit详细搭建过程(转)
2015-06-01 09:46 3593目录[-] springMVC+mybatis框 ... -
linux awk的使用详解(收藏)
2014-09-03 16:04 710简介 awk是一个强大的文本分析工具,相对于grep的查 ... -
JSP中文乱码的产生原因及解决方案-转
2013-05-15 17:33 2413JSP中文乱码的产生原因及解决方案在JSP的开发过程中,经常 ... -
SAML介绍
2013-03-25 21:03 4427工作了一段时间,接触到不少知识,好久没有进行总 ... -
逆向工程中的问题:Hibernate project not selected 解决方式(转)
2012-03-14 10:43 2191今天用myeclipse自动生成POJO类时一直找不到java ... -
责任链模式(转)
2011-11-22 21:38 1092责任链模式是一种对象的行为模式。 在责任链模 ... -
Java序列化与反序列化
2011-11-08 23:38 566基本概念:序列化是将对象状态转换为可保持或传输的格式的过程。与 ... -
java动态代理--JDK
2011-10-30 18:04 942JAVA的动态代理 代理模式 代理模式是常用的java ... -
Linux环境变量的设置和查看方法
2011-10-23 21:40 1745Linux是一个多用户的操作系统。每个用户登录系统后,都 ... -
Hibernate的merge与update方法的区别
2011-09-20 22:47 984今天做了个测试,写了个测试用例来看看merge与update时 ... -
Hibernate学习:Transient、Persistent、Detached三个状态
2011-09-19 22:15 1448Transient、Persistent、Detached是H ... -
sping源码分析之properties操作(转载) .
2011-09-15 14:36 1183目前在开发的过程中遇到:通过spring在后台对前台 ... -
Annotation 介绍
2011-07-07 10:29 1219从JDK5开始提供名为Annotation(注释)的功能,它被 ... -
java子类对象的初始化过程(转)
2010-12-13 10:28 1058具体的顺序如下: ... -
Linux创建目录与删除目录命令具体分析
2010-12-13 09:40 4079最近被人问到如何在linux下创建多层目录结构,当时只是知道m ... -
策略模式学习
2010-12-09 22:27 997一个策略模式的加减乘除: 抽象策略角色: (精囊妙计) ...
相关推荐
标题中的"java写的一个使用jni调用c/c++的dll"意味着我们要创建一个Java项目,该项目包含一个或多个`native`方法,这些方法的实现将在C/C++中完成。接着,我们需要使用`javah`工具生成JNI头文件,这个头文件定义了...
JNI提供了一套接口,让Java虚拟机(JVM)能够调用本地方法,这些方法由C或C++编写,并编译成动态链接库(如Windows下的.dll或Linux下的.so文件)。JNI框架包括了Java端的本地方法声明和本地方法实现,以及C/C++端的...
在本主题中,我们将深入探讨如何在Windows和Linux环境中利用JNI来生成动态链接库(DLL for Windows,SO for Linux),并使Java能够调用C++编写的函数。 1. **Java调用C++的基本原理** - JNI为Java应用程序提供了一...
本教程将详细介绍如何使用C/C++实现Java的Native方法接口,并通过代码实例来展示JNI编程的基本步骤。 1. **JNI简介** JNI是一种规范,它定义了Java虚拟机(JVM)和本地代码之间的接口。通过JNI,开发者可以在Java...
使用C/C++编译器(如GCC或Visual C++)编译源代码,并链接到Java运行时库。创建DLL文件,例如`myDll.dll`。 6. **运行Java程序** 在Java程序中,通过`System.loadLibrary`加载生成的DLL,然后就可以调用C/C++实现...
Java程序可以通过JNI调用本地方法,这些本地方法是用其他语言编写的,并通过JNI接口暴露给Java。这个过程涉及到以下几个步骤: 1. **创建本地方法声明**:在Java类中,你需要声明本地方法。这些方法没有具体实现,...
本资源详细介绍了如何在Linux环境中通过C语言使用JNI调用Java类的函数,以下是对这个主题的详细阐述。 首先,要进行C语言调用Java类,我们需要确保有Java Development Kit (JDK) 安装并正确配置。JDK包含了一系列...
本文将深入探讨如何使用C++创建Java虚拟机(JVM),并通过JNI来调用Java函数。这是一项技术性很强的任务,需要对C++编程、Java虚拟机的工作原理以及JNI接口有深入的理解。 首先,Java虚拟机(JVM)是Java平台的核心...
Java通过JNI(Java Native Interface)调用C语言函数库是一种常见的技术实践,它允许Java程序直接与本地操作系统和硬件交互,提高性能或者利用已有的C/C++代码库。本资料"Java通过JNI调用C语言函数库的方法.zip"提供...
在这种情况下,"springboot+jna/jni调用动态so/dll库"是一个重要的主题,它涉及到Spring Boot应用如何利用Java Native Interface (JNI) 和 Java Native Access (JNA) 这两种技术来调用操作系统级别的动态链接库(.so...
这个“testjni.rar”压缩包包含了一个关于如何使用C或C++调用Java方法的示例。让我们深入了解一下这个过程以及相关知识点。 首先,JNI是Java平台的一部分,它为Java开发者提供了一种方式来编写本地方法,这些方法...
本示例DEMO将详细介绍如何在Java程序中通过JNI调用C++代码函数。 首先,我们需要理解JNI的工作原理。JNI提供了一种方式,让Java虚拟机(JVM)能够调用本地方法,这些方法可以是C、C++或者其他本地语言编写的。JNI...
Java调用C/C++是跨语言编程中常见的一种技术,主要通过Java的Native Interface (JNI) 实现。JNI 是Java平台标准的一部分,它允许Java代码和其他语言写的代码进行交互,尤其是C和C++。在很多情况下,这可以利用现有的...
JNI是一种Java API,它定义了一组规则,允许Java代码调用本地方法(即非Java编写的代码),同时也允许本地代码调用Java方法。这些本地方法通常以库文件的形式存在(如Windows下的.dll或Unix/Linux下的.so文件)。 #...
为了弥补这一缺陷,Java引入了JNI(Java Native Interface)技术,允许Java代码直接调用本地C/C++代码,从而实现更深层次的操作系统访问和性能优化。 #### 二、Java调用C/C++的基础概念 在讨论具体过程之前,我们...
JNI提供了一种机制,使得Java应用程序能够调用非Java编写的函数,这些函数通常存储在动态链接库(Windows下的.dll文件或者Unix/Linux下的.so文件)中。下面将详细介绍如何使用JNI来调用C/C++代码。 #### 三、JNI...
本文的标题“Java 通过 JNI 调用 C++ 的 DLL 文件”表明了文章的主要内容,即使用 JNI 将 Java 应用程序与 C++ DLL 文件集成。 描述解析 文章的描述提供了一个 Demo 文档,描述了如何使用 JNI 将 Java 应用程序与 ...
### Java JNI调用动态库(Linux、Windows)的实现步骤 ...通过以上步骤,您就可以成功地在 Java 应用中使用 C/C++ 编写的本地方法了。这对于提高性能、利用现有 C/C++ 代码库或访问特定平台功能都非常有用。
在这里,`extern "C"`用于告诉编译器使用C语言的链接规则,`JNIEXPORT`和`JNICALL`是JNI定义的宏,用于指定函数的导出和调用约定。`JNIEnv *env`参数是JNI接口指针,用于访问JNI函数。`jobject`参数是Java对象的引用...