c++调用java

1. JNIEnv对象 

对于本地函数

JNIEXPORT void JNICALL Java_video1_TestNative_sayHello(JNIEnv * env, jobject obj)

{

cout<<"Hello Native Test !"<<endl;

}

JNIEnv类型代表Java环境。通过这个JNIEnv*指针,就可以对Java端的代码进行操作。如,创建Java类得对象,调用Java对象的方法,获取Java对象的属性等。

JNIEnv的指针会被JNI传送到本地方法的实现函数中来对Java端的代码进行操作

JNIEnv类中的函数:

NewObject/NewString/New<TYPE>Array  :new新对象

 Get/Set<TYPE>Field:获取属性

 Get/SetStatic<TYPE>Field :获取静态属性

Call<TYPE>Method/CallStatic<TYPE>Method:调用方法

2. Java数据类型与C/C++数据类型的对应关系

可以参考 jni.h 文件:http://home.pacifier.com/~mmead/jni/cs510ajp/jni.h

Java类型      别名             C++本地类型          字节(bit)  
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

3. 获取jclass

为了能够在C/C++使用Java类,jni.h头文件中专门定义了jclass类型来表示Java中的Class类

jclass的取得:

JNIEnv类中有如下几个简单的函数可以取得jclass

jclass FindClass(const char* clsName)  根据类名来查找一个类,完整类名。

jclass GetObjectClass(jobject obj)   根据一个对象,获取该对象的类

jclass GetSuperClass(jclass obj)     获取一个类的父类

FindClass 会在classpath系统环境变量下寻找类,需要传入完整的类名,注意包与包之间是用"/"而不是"."来分割

如:jclass cls_string= env->FindClass("java/lang/String");

获取jclass又什么用,比如你要调用类的静态方法,静态属性就需要通过这个方法来获取一个类。

4. 本地代码访问Java类中的属性与方法 

有了类和对象之后,如何才能访问java中的对象的属性和方法呢,这就需要用到以下这些方法了。

JNI在jni.h头文件中定义了jfieldID,jmethodID类表示Java端的属性和方法

如何获取属性: 在访问或设置Java属性的时候,首先就要现在本地代码中取得代表Java属性的jfieldID,然后才能在本地代码中进行Java属性操作。

如何调用java的方法:调用Java端的方法时,需要取得代表方法的jmethodID才能进行Java方法调用

JNIEnv获取相应的fieldID和jmethodID的方法:

GetFieldID/GetMethodID

GetStaticFieldID/GetStaticMethodID

GetMethodID也可以取得构造函数的jmethodID。创建Java对象时调用指定的构造函数。

如:env->GetMethodID(data_Clazz,"method_name","()V")

(*jniEnv)->GetMethodID(jniEnv, Clazz,"<init>", "()V");

这个比较特殊,这个是默认构造函数的方法,一般用这个来初始化对象,但是再实际过程中,为了快速生成一个实例,一般通过工厂方法类创建jobject

jni.h 对GetMethodID的定义:

jmethodID (JNICALL *GetMethodID)

(JNIEnv *env, jclass clazz, const char *name, const char *sig);

这就引入了一个新的问题,什么是sig,我们后面再说,举个例子说明

前提说明: JAVA类 TestProvider ,该类有2个方法分别为String getTime( ) , void saysayHello( String str)

jclass TestProvider;
jobject mTestProvider;
jmethodID getTime;
jmethodID sayHello;

C 中映射类

TestProvider = (*jniEnv)->FindClass(jniEnv,"com/duicky/TestProvider");

C中新建对象

//默认构造函数,不传参数
       jmethodID construction_id = (*jniEnv)->GetMethodID(jniEnv, TestProvider,"<init>", "()V");
       //通过NewObject来创建对象
       jobject mTestProvider = (*jniEnv)->NewObject(jniEnv, TestProvider,construction_id);

C 中映射方法

静态:
getTime = (*jniEnv)->GetStaticMethodID(jniEnv, TestProvider, "getTime","()Ljava/lang/String;");
       非静态:
sayHello = (*jniEnv)->GetMethodID(jniEnv, TestProvider, "sayHello","(Ljava/lang/String;)V");

C 中调用 Java的 方法

静态:
(*jniEnv)->CallStaticObjectMethod(jniEnv, TestProvider, getTime);
       非静态:
(*jniEnv)->CallVoidMethod(jniEnv, mTestProvider, sayHello,jstrMSG);

注意 GetXXXMethodID  和 CallXXXMethod 。

第一个XXX 表示的是映射方法的类型,如: 静态 跟非静态

第二个 XXX 表示 调用方法的返回值 ,如:Void,Object,等等。(调用静态方法的时候Call后面要加Static)

  5. sign签名

对于 jmethodID GetMethodID(jclass clazz, const char *name, const char *sign)

clazz代表该属性所在的类,name表示方法名称,sign是签名

那什么是签名,签名是对函数参数和返回值的描述,对同一个函数,在java中允许重载,这个时候就需要这个sign来进行区分了。

以下是java类型签名的描述

用来表示要取得的属性/方法的类型

类型           相应的签名  
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类型签名···)返回值类型签名

特别注意:Object后面一定有分号(;)结束的,多个对象参数中间也用分号(;)来分隔

例子:

方法签名

void f1()                         ()V
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

图解签名:

使用javap命令来产生签名

javap -s -p [full class Name]

-s 表示输出签名信息

-p 同-private,输出包括private访问权限的成员信息

例子:

C:\E\java\workspaces\myeclipseblue\JNITest\bin>javap -s -private video1.TestNative  
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完整代码:

package video1;  
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 "video1_TestNative.h"  
#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  
 }

本文地址,转载请注明出处:

http://www.cnblogs.com/likwo/archive/2012/05/21/2512400.html

参考资料:

http://zzqrj.iteye.com/blog/1285262

jni.h 头文件:

http://home.pacifier.com/~mmead/jni/cs510ajp/jni.h

相关例子:

http://www.pacifier.com/~mmead/jni/cs510ajp/index.html

Programmming in C/C++ with the Java Native Interface (3 个练习)

http://www.pacifier.com/~mmead/jni/cs510ajp/exercises/index.html

JNI 文档:

http://files.cnblogs.com/luxiaofeng54/JNI_Docs.rar

基于 Android NDK 的学习之旅----- C调用Java

http://www.cnblogs.com/luxiaofeng54/archive/2011/08/17/2142000.html

Linux下JNI的使用:比较基础

http://www.cnblogs.com/bastard/archive/2012/05/17/2506877.html

如何在Android下使用JNI:讲解比较详细,但是代码里有些错误,空格没处理好

http://blog.csdn.net/xnwyd/article/details/7086384

这篇文章有些地方不清楚的参考下这篇文章

Android Jni代码示例讲解

http://developer.51cto.com/art/201001/181355.htm

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 开发

使用 Java Native Interface 的最佳实践:描述了JNI性能和缓存的一些东西

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

时间: 2024-07-29 15:43:52

c++调用java的相关文章

基于 Android NDK 的学习之旅----- C调用Java

http://www.cnblogs.com/luxiaofeng54/archive/2011/08/17/2142000.html 基于 Android NDK 的学习之旅----- C调用Java许多成熟的C引擎要移植到Android 平台上使用 , 一般都会 提供 一些接口, 让Android sdk 和 jdk 实现. 下文将会介绍 C 如何 通过 JNI 层调用 Java 的静态和非静态方法. 1.主要流程 1.  新建一个测试类TestProvider.java a)       

如何在android平台上使用js直接调用Java方法[转]

转载自:http://www.cocos.com/docs/html5/v3/reflection/zh.html #如何在android平台上使用js直接调用Java方法 在cocos2d-js 3.0beta中加入了一个新特性,在android平台上我们可以通过反射直接在js中调用java的静态方法.它的使用方法很简单: var o = jsb.reflection.callStaticMethod(className, methodName, methodSignature, parame

Android的WebView通过JS调用java代码

做项目时候会遇到我们用WebView 打开一个web,希望这个web可以调用自己的一些方法,比如我们在进一个web页面,然后当我们点击web上的某个按钮时,希望能判断当前手机端是否已经登录,如果未登录,那么就会跳转到登录页面(登陆页面是另一个Activity).这个时候,一个简单的做法就是在按钮动作事件的js上调用java的方法,从而起到判断是否登录,并决定是否跳转到另一个页面. Google的WebView为我们提供了 addJavascriptInterface(Object obj, St

hessian 通信,C#客户端调用java服务端服务

Hessian是一个轻量级的remoting onhttp工具,使用简单的方法提供了RMI的功能. 相比WebService,Hessian更简单.快捷.采用的是二进制RPC协议,因为采用的是二进制协议,所以它很适合于发送二进制数据(来自百度) 本次通过hessian简单实现C#调用java服务,并返回java方法的结果,其中C#以及java方法的参数可以是Entity. 1. 依赖 java包及DLL   1.1 hessian-4.0.37.jar   1.2 Hessiancsharp.d

LuaJ 调用java方法性能研究

先简单写一下: lua调用java的方法,在luaJ中有两种. 1 使用官方提供的luaJava库 local ins = luajava.newInstance( "com.test.lj.TestClass");ins:test() 2 自己绑定方法,采用类似原版lua的方式将方法 local ins = TestClass.new() --new方法自己绑定 ins:test() 哪种方法好? 结论: 令人惊讶的是,luajava库反射调用的性能在某些情况下,竟然比java原生绑

C#调用Java方法

C#调用Java方法(详细实例) 阅读目录 C#调用c++ C#调用JAVA方法 C#可以直接引用C++的DLL和转换JAVA写好的程序.最近由于工作原因接触这方面比较多,根据实际需求,我们通过一个具体例子把一个JAVA方法转换成可以由C#直接调用的DLL 回到目录 C#调用c++ C#调用C++的例子网上很多,以一个C++的具体方法为例. C++代码 // 获取一帧图像数据 MVSMARTCAMCTRL_API int __stdcall MV_SC_GetOneFrame(IN void*

DelphiXE7如何调用Java Class,JAR等文件?

源文地址:http://jingyan.baidu.com/article/e4d08ffdb61b040fd3f60d44.html 第一步,我们先在互联网上把java2pas这个工具下载下来. 下载地址:http://download.csdn.net/detail/zqycn/7791619 下载地址:http://download.csdn.net/detail/zhujianqiangqq/8985597 第二步,打开[windows菜单]-->在运行框中输入[cmd]敲回车键. 如:

jni中调用java方法获取当前apk的签名文件md5值

相应的java方法: void getsign(Context context) throws Exception { PackageInfo localPackageInfo = context.getPackageManager() .getPackageInfo(context.getPackageName(), 64); StringBuilder localStringBuilder = new StringBuilder(); Signature[] arrayOfSignature

【cocos2d-js官方文档】二十四、如何在android平台上使用js直接调用Java方法

在cocos2d-js 3.0beta中加入了一个新特性,在android平台上我们可以通过反射直接在js中调用java的静态方法.它的使用方法很简单: var o = jsb.reflection.callStaticMethod(className, methodName, methodSignature, parameters...) 在callStaticMethod方法中,我们通过传入Java的类名,方法名,方法签名,参数就可以直接调用Java的静态方法,并且可以获得Java方法的返回

Esper调用Java代码执行系统命令

Esper语言是可以调用高级语言的静态方法的: 以调用Java的静态方法执行系统命令为例: select Runtime.getRuntime().exec(\"calc\"),avg(price) from " + product + ".win:length_batch(3) 完整代码可以参考如下链接: http://blog.csdn.net/luonanqin/article/details/9900295