AndroidJNI 通过C++调用JAVA

1. JNIEnv对象    对于本地函数   JNIEXPORT void JNICALL Java_video1_TestNative_sayHello(JNIEnv * env, jobject obj)   {        cout   }           JNIEnv类型代表Java环境。通过这个JNIEnv*指针,就可以对Java端的代码进行操作。如,创建Java类得对象,调用J

大家好,又见面了,我是全栈君。

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
           3
2  


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);
 
注意 
Get
XXX
MethodID  和 
Call
XXX
Method 。
第一个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
 
 

 图解签名:

AndroidJNI 通过C++调用JAVA 

 

 
使用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 

 

参考资料:

 

 
jni.h 头文件:

 

相关例子:

 

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

 

 
JNI 文档:

 

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

 

 
Linux下JNI的使用:比较基础

 

 
如何在Android下使用JNI:讲解比较详细,但是代码里有些错误,空格没处理好
这篇文章有些地方不清楚的参考下这篇文章
 
Android Jni代码示例讲解

 

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 

本文转自http://www.cnblogs.com/likwo/archive/2012/05/21/2512400.html

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/121338.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(1)


相关推荐

  • 微信公众平台开发者社区_php微擎框架

    微信公众平台开发者社区_php微擎框架一、思考开发了几个微信项目,一直在思考:如何将微信相关的处理与业务系统联系在一起?如何做到彼此分离,且易于扩展?能否开发一套独立的微信服务框架,支持各种业务应用?二、现有常用的服务框架支持多种业务应用,我们通过分层的方式来实现。将复杂的系统进行分层,将一些功能或者特有的逻辑进行封装,封装为不同的基础服务或中间件。业务层无需关心底层具体实现,只需进行简单调用、组装

  • verilog 移位寄存器与流水灯「建议收藏」

    verilog 移位寄存器与流水灯「建议收藏」用Verilog写一个简单的多功能移位寄存器,并在移位寄存器的基础上进行修改,利用移位功能形成流水灯的效果基本的功能如下:寄存器示意图见下方虚拟面板图有两种改进方式:第一种是把最后一个功能改成循环右移,即此功能模式下,按一下时钟按键(代表时钟信号到来),高三位往右移一位,最低位移动到最高位。按一次移位一次,形成流水灯效果第二种在第一种的基础上把按键时钟改成接入内部时钟,每隔一段时间时钟…

  • Redis的配置文件详解「建议收藏」

    Redis的配置文件详解「建议收藏」目录1、开头说明2、INCLUDES3、MODULES4、NETWORK5、GENERAL6、SNAPSHOTTING7、REPLICATION8、SECURITY9、CLIENTS10、MEMORYMANAGEMENT11、APPENDONLYMODE12、LUASCRIPTING13、REDISCLUSTER资料推荐   上一篇博客我们介绍了如何安装Redis,在Redis的解压目录下有个很重要的配置文件redis.conf (/opt/redis-4.0.9目录下),

  • SSRF漏洞分析与实践

    SSRF漏洞分析与实践警告请勿使用本文提到的内容违反法律。本文不提供任何担保目录警告一、漏洞简介二、漏洞产生的原因三、SSRF的类型四、利用SSRF可以实现的攻击五、实践漏洞学习(一)简单漏洞验证(二)通过dict协议获取服务器端口运行的服务。(三)通过file协议访问计算机中的任意文件(四)了解sftp及tftp协议在SSRF中的作用六、SSRF与CSRF的区别七、漏洞修复警告一、漏洞简介SSRF(Server-SideRequest…

  • sdfsfd

    sdfsfd

    2021年11月14日
  • SSM项目(GitHub上找的)

    SSM项目(GitHub上找的)SSM项目1.学生信息管理系统链接:https://pan.baidu.com/s/1e9ar4OKetL-40mp6R0b_4w提取码:01c8运行环境:jdk1.8以上服务器:tomcat运行软件:eclipse界面如下2.学生考试系统运行环境:jdk1.8以上服务器:tomcat运行软件:eclipse2.1学生前台2.2后台3.房屋出租系统运行环境:jdk1.8以上服务器:tomcat运行软件:eclipse…

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号