Android C++和JAVA互相调用
来源:互联网 发布:java内嵌机制 编辑:程序博客网 时间:2024/05/18 01:55
boolean jboolean unsigned char 8, unsigned
byte jbyte signed char 8
char jchar unsigned short 16, unsigned
short jshort short 16
int jint long 32
long jlong __int64 64
float jfloat float 32
double jdouble double 64
void void n/a
Object _jobject *jobject
jobject mTestProvider;
jmethodID getTime;
jmethodID sayHello;
C中新建对象
jmethodID construction_id = (*jniEnv)->GetMethodID(jniEnv, TestProvider,"<init>", "()V");
//通过NewObject来创建对象
jobject mTestProvider = (*jniEnv)->NewObject(jniEnv, TestProvider,construction_id);
getTime = (*jniEnv)->GetStaticMethodID(jniEnv, TestProvider, "getTime","()Ljava/lang/String;");
非静态:
sayHello = (*jniEnv)->GetMethodID(jniEnv, TestProvider, "sayHello","(Ljava/lang/String;)V");
(*jniEnv)->CallStaticObjectMethod(jniEnv, TestProvider, getTime);
非静态:
(*jniEnv)->CallVoidMethod(jniEnv, mTestProvider, sayHello,jstrMSG);
boolean Z
byte B
char C
short S
int I
long J
float F
double D
void V
object L用/分隔包的完整类名: Ljava/lang/String;
Array [签名 [I [Ljava/lang/Object;
Method (参数1类型签名 参数2类型签名···)返回值类型签名
int f2(int, long) (IJ)I
boolean f3(int[]) ([I)B
double f4(String, int) (Ljava/lang/String;I)D
void f5(int, String [], char) (I[Ljava/lang/String;C)V
图解签名:
Compiled from "TestNative.java"
public class video1.TestNative extends java.lang.Object{
public java.lang.String name;
Signature: Ljava/lang/String;
public video1.TestNative();
Signature: ()V
public int signTest(int, java.util.Date, int[]);
Signature: (ILjava/util/Date;[I)I
public native void sayHello();
Signature: ()V
public static void main(java.lang.String[]);
Signature: ([Ljava/lang/String;)V
}
TestNative完整代码:
import java.util.Date;
public class TestNative {
public String name="Test";
public int number =100;
public int signTest(int i,Date date,int[] arr){
System.out.println("Sign Test");
return 0;
}
//native关键字修饰的方法,其内容是C/C++编写的,java中不必为它编写具体的实现
public native void sayHello();
public static void main(String[] args) {
System.loadLibrary("NativeCode");
TestNative tn = new TestNative();
tn.sayHello();
}
}
C/C++代码
#include <iostream>
using namespace std;
JNIEXPORT void JNICALL Java_video1_TestNative_sayHello(JNIEnv * env, jobject obj){
cout<<"Hello Native Test !"<<endl;
//因为test不是静态函数,所以传进来的就是调用这个函数的对象
//否则就传入一个jclass对象表示native()方法所在的类
jclass native_clazz = env->GetObjectClass(obj);
//得到jfieldID
jfieldID fieldID_prop = env->GetFieldID(native_clazz,"name","Ljava/lang/String;");
jfieldID fieldID_num = env->GetFieldID(native_clazz,"number","I");
//得到jmethodID
jmethodID methodID_func=env->GetMethodID(native_clazz,"signTest","(ILjava/util/Date;[I)I");
//调用signTest方法
env->CallIntMethod(obj,methodID_func,1L,NULL,NULL);
//得到name属性
jobject name = env->GetObjectField(obj,fieldID_name);
//得到number属性
jint number= env->GetIntField(obj,fieldID_num);
cout<<number<<endl;//100
//修改number属性的值
env->SetIntField(obj,fieldID_num,18880L);
number= env->GetIntField(obj,fieldID_num);
cout<<number<<endl;//18880
}
摘要:
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 {
}
编译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)
{
}
char* jstringTostring(JNIEnv* env, jstring jstr)
{
}
using namespace std;
int main()
{
}
编译该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:
#
#
#
# Java VM: Java HotSpot(TM) Server VM (1.5.0_11-b03 mixed mode)
。。。。。
参考文档:使用 Java Native Interface 的最佳实践
本文地址,转载请注明出处:
http://www.cnblogs.com/likwo/archive/2012/05/21/2512400.html
JNI callMethod参考文档
其他推荐学习网站
JNI的提高,Java类型和C(C++)类型转换源代码
http://blog.csdn.net/ostrichmyself/article/details/4557851
JNI 的多线程
http://blog.csdn.net/popop123/article/details/1511180
Android NDK 开发
https://www.ibm.com/developerworks/cn/java/j-jni/
JNI 攻略系列
JNI全攻略之一--建立一个简单的JNI程序
http://blog.csdn.net/yjkwf/article/details/7006260
JNI全攻略之二――JNI基础
http://blog.csdn.net/yjkwf/article/details/7006261
JNI全攻略之三--JNI头文件分析
http://blog.csdn.net/yjkwf/article/details/7006264
JNI攻略之四――JNI操作数组
http://blog.csdn.net/yjkwf/article/details/7006266
http://disanji.net/2011/01/26/android-jni-programming-2/
JNI Examples for Android
http://android.wooyd.org/JNIExample/files/JNIExample.pdf
JNI pthread 多线程使用
http://www.cnblogs.com/lknlfy/archive/2012/03/16/2400786.html
- Android JNI之Java和C互相调用
- Android WebView java和js互相调用
- android js 和java互相调用
- Android C++和JAVA互相调用
- Android JNI中C和JAVA代码之间的互相调用
- C++ 和 C 互相调用
- Android NDK开发 Java与C互相调用实例详解
- Android NDK开发----- Java与C互相调用实例详解
- Android NDK开发----- Java与C互相调用实例详解
- Android Ndk中C与JAVA之间的互相调用
- JNI学习(一)(c和java层对象互相调用)
- Android和JavaScript互相调用
- Android和JavaScript互相调用
- Android和JavaScript互相调用
- Android和JavaScript互相调用
- Android 和 js 互相调用
- Android和JavaScript互相调用
- Android和JavaScript互相调用
- 2017年8月14日提高组T2 温度
- C++ 顺序容器
- 单调栈
- 九度1439:Least Common Multiple
- 使用Spring报错:No default constructor found;
- Android C++和JAVA互相调用
- Android Studio 优秀插件汇总
- HBase与Hive整合的必要性
- 折半枚举(双向搜索)poj27854 Values whose Sum is 0
- (NOIP2015第二题)扫雷(mine)
- 初学常用命令(二)
- Linux设备树, .dtb文件
- 敌兵布阵||HDU1166
- poj 2533 最长上升子序列 多种姿势