C++ 跨语言调用 Java  

  Java JDK 提供了 JNI 接口供 C/C++ 程序调用 Java 编译后的类与方法,主要依赖于头文件(jni.h) 和 动态库(jvm.so/jvm.dll),由于 JNI 包含了丰富的接口映射和跨语言的数据通信,非常复杂(坑 深不见底),所以这里只对一个测试程序进行简单的描述。

  最开始测试的时候选择了 Window7 64 的环境,安装的 Java JDK 也是64位的,但是我们都知道 VS 编译的程序默认情况下都是32位程序,所以我在 LoadLibrary(“jvm.dll”)的时候总是失败,所以就放弃了 Windows 环境下的测试(懒得去编译64的程序),最终我使用了 CentOS7 64 完成了测试,并且测试的 Java 版本是 1.7(系统自带openjdk)。

测试的 Java 代码如下所示。

 public class MyTest {
private static int magic_counter = 777; public static void callback() {
System.out.println("Hello world in java from cplusplus");
System.out.print("Magic number: ");
System.out.println(magic_counter);
}
}

  编写完 Java 测试程序之后,使用命令:javac MyTest.java 对 Java 类进行编译,并生成相应的 MyTest.class 文件,这个文件所在的位置非常重要,这关系到我们的 C++ JNI 程序能否找到这个文件,因为我们在 C++ JNI 程序中指定了 "-Djava.class.path=." 就是C++程序运行的当前目录,所有这里我们需要把 MyTest.class 文件拷贝到 C++ 程序的运行目录中。

  此外由于 JNI 函数需要将调用对象的 Signature ID 传入,所以我们还需要知道你所有使用对象的 Signature。通过命令: javap -s -p MyTest.class 命令可以获取后所有函数与变量的 Signature。如下图所示。

  在编译程序之前,需要了解你机器上的 Java 版本,及 jni.h 和 jvm.so 所在的位置,方便程序编译时能够找到相应的 Java 依赖,这里我使用了 CMake 来辅助编译,相应的 CMakeList.txt 如下所示。

cmake_minimum_required(VERSION 3.5)
project(testjni) set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11") include_directories(
#/opt/program/jdk1..0_121/include/linux
#/opt/program/jdk1..0_121/include
/usr/lib/jvm/java-1.7.-openjdk-1.7.0.111-2.6.7.2.el7_2.x86_64/include/linux
/usr/lib/jvm/java-1.7.-openjdk-1.7.0.111-2.6.7.2.el7_2.x86_64/include )
link_directories(
#/opt/program/jdk1..0_121/jre/lib/amd64/server
/usr/lib/jvm/java-1.7.-openjdk-1.7.0.111-2.6.7.2.el7_2.x86_64/jre/lib/amd64/server/
) set(SOURCE_FILES main.cpp)
add_executable(testjni ${SOURCE_FILES}) target_link_libraries(
testjni
jvm
)

  下面是 C++ JNI 程序的简单示例,如下所示。

#include <iostream>
#include <jni.h>
#include <memory.h> int main()
{
char opt1[] = "-Djava.compiler=NONE"; /** 暂时不知道啥意思,网上抄来的 */
char opt2[] = "-Djava.class.path=."; /** 指定Java类编译后.class文件所在的目录 */
char opt3[] = "-verbose:NONE"; /** 暂时不知道啥意思,网上抄来的 */ JavaVMOption options[];
options[].optionString = opt1; options[].extraInfo = NULL;
options[].optionString = opt2; options[].extraInfo = NULL;
options[].optionString = opt3; options[].extraInfo = NULL; JavaVMInitArgs jargv;
jargv.version = JNI_VERSION_1_6; /** JDK JNI VERSION*/
jargv.nOptions = ;
jargv.options = options;
jargv.ignoreUnrecognized = JNI_TRUE; JavaVM* jvm = NULL;
JNIEnv* jenv = NULL;
jint res = JNI_CreateJavaVM( &jvm, (void**)&jenv, &jargv );
if ( != res )
return ; jclass jc = jenv->FindClass( "MyTest" );
if ( NULL == jc )
return ; jmethodID jmid = jenv->GetStaticMethodID( jc, "callback", "()V" );
if ( NULL == jmid )
return ; jenv->CallStaticVoidMethod( jc, jmid ); /** 在网上没有找到任何关于空间相关 JavaVM 和 JNIEnv 资源释放的描述 */
std::cout << "Hello, World!" << std::endl;
return ;
}

  最终输出的结果与预期的一致,就到这里,在这么深的坑里,祝大家好运哦。

C++ 跨语言调用 Java的更多相关文章

  1. Atitit java c# php c++ js跨语言调用matlab实现边缘检测等功能attilax总结

    Atitit java c# php c++ js跨语言调用matlab实现边缘检测等功能attilax总结 1.1. 边缘检测的基本方法Canny最常用了1 1.2. 编写matlab边缘检测代码, ...

  2. Golang、Php、Python、Java基于Thrift0.9.1实现跨语言调用

    目录: 一.什么是Thrift? 1) Thrift内部框架一瞥 2) 支持的数据传输格式.数据传输方式和服务模型 3) Thrift IDL 二.Thrift的官方网站在哪里? 三.在哪里下载?需要 ...

  3. Java跨语言调用,使用JNA访问Java外部接口

    1. JNA简单介绍 先说JNI(Java Native Interface)吧,有过不同语言间通信经历的一般都知道,它允许Java代码和其他语言(尤其C/C++)写的代码进行交互,只要遵守调用约定即 ...

  4. 使用thrift进行跨语言调用(php c# java)

    使用thrift进行跨语言调用(php c# java)   1:前言 实际上本文说的是跨进程的异构语言调用,举个简单的例子就是利用PHP写的代码去调C#或是java写的服务端.其实除了本文提供的办法 ...

  5. Golang通过Thrift框架完美实现跨语言调用

    每种语言都有自己最擅长的领域,Golang 最适合的领域就是服务器端程序. 做为服务器端程序,需要考虑性能同时也要考虑与各种语言之间方便的通讯.采用http协议简单,但性能不高.采用TCP通讯,则需要 ...

  6. vs2019 Com组件初探-简单的COM编写以及实现跨语言调用

    前提条件 1.掌握C++基础语法 2.平台安装 vs2019 3.本地平台为 windows 10 1909 X64 4.了解vbs基础语法 本次目标 1.掌握Com组件的概念及原理 2.编写一个简单 ...

  7. 跨语言调用Hangfire定时作业服务

    跨语言调用Hangfire定时作业服务 背景 Hangfire允许您以非常简单但可靠的方式执行后台定时任务的工作.内置对任务的可视化操作.非常方便. 但令人遗憾的是普遍都是业务代码和hagnfire服 ...

  8. 03_Android NDK中C语言调用Java代码,javah的使用,javap的使用以及生成签名,Android.mk的编写,C代码的编写

     1  案例场景,通过C语言回调Java的代码,案例的最终界面: 2  案例的代码结构如下: 3 编写DataProvider的代码: package com.example.ndkcallbac ...

  9. CLS(公共语言规范)的CLSCompliant(跨语言调用)

    .net的一个很重要的特性就是跨语言的编程,用C#写的dll可以在VB.net里调用,例如:用C#写的一个类,编译到dll中,然后在VB.net中调用: using System;namespace  ...

随机推荐

  1. Python select实现socket并发

    Python select  Python的select()方法直接调用操作系统的IO接口,它监控sockets,open files, and pipes(所有带fileno()方法的文件句柄)何时 ...

  2. Python Redis 管道

    redis-py默认在执行每次请求都会创建(连接池申请连接)和断开(归还连接池)一次连接操作,如果想要在一次请求中指定多个命令,则可以使用pipline实现一次请求指定多个命令,并且默认情况下一次pi ...

  3. JavaScript形而上的策略模式

    什么是策略模式? 先看代码片段1. // 代码片段1 var bonus = new Bonus(); bonus.setSalary(10000); bonus.setStrategy(new pe ...

  4. 深入理解Plasma(二)Plasma 细节

    这一系列文章将围绕以太坊的二层扩容框架,介绍其基本运行原理,具体操作细节,安全性讨论以及未来研究方向等.本篇文章主要对 Plasma 一些关键操作的细节进行剖析. 在上一篇文章中我们已经理解了什么是 ...

  5. DFS例题

    特殊的质数肋骨(递归)] -题目描述-农民约翰的母牛总是生产出最好的肋骨.你能通过农民约翰和美国农业部标记在每根肋骨上的数字认出它们. 农民约翰确定他卖给买方的是真正的质数肋骨,是因为从右边开始切下肋 ...

  6. (02) 第一个springboot程序

    1. 创建一个springboot程序 1. idea 自带的springboot插件 2. 直接从https://start.spring.io 创建好程序下载下来, 之后覆盖你的创建的项目 2. ...

  7. Linux获取so/ko文件版本号教程

    一.需要获取版本号的原因 从使用角度而言,有时只有特定版本的库才支持某些功能,所以我们需要确定库文件版本号. 从安全加固角度而言,有些版本存在漏洞有些版本不存在漏洞,所以我们需要获取版本号以确定当前使 ...

  8. react-router解决锚点跳转问题

    添加一个onClick方法.onClick方法传入一个锚点的id,然后用下面的函数来找到锚点并跳转到锚点. scrollToAnchor = (anchorName) => { if (anch ...

  9. Servlet过滤器实现访客人数统计

    第一. Servlet的创建和配置  1. 创建一个Servlet需要实现javax.servlet.Filter接口,同时实现Filter的3个方法.             第一个方法时过滤器中的 ...

  10. [Codeforces477D]Dreamoon and Binary

    Problem 给定一个字符串数的二进制表示(不含前导0)s(长度不超过5000), 对于一个数n(初值为0),可以进行以下两种操作: 1.将n的二进制表示(无前导0)写到已经写的串的后面. 2.n加 ...