AndroidJNI 调用JAVA(转)-程序员宅基地

技术标签: java  移动开发  c/c++  

转自:http://www.cnblogs.com/likwo/archive/2012/05/21/2512400.html
 
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
 
 

 图解签名:

 

 

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

转载于:https://www.cnblogs.com/colife/p/3498090.html

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/weixin_34221775/article/details/93949984

智能推荐

仿微博@与#话题#功能的可折叠的TextView_java 话题功能-程序员宅基地

文章浏览阅读1k次。找了很多篇博客, 都没看到我自己想要的那种效果, 就自己写了一个效果图使用第一步:添加如下代码至project的build.gradle里allprojects { repositories { ... maven { url 'https://jitpack.io' } } }第二步:在module的build.gradle添加depende......_java 话题功能

java枚举使用_java 枚举 键值对-程序员宅基地

文章浏览阅读435次。在C++时代就有了枚举值这个类型,它是一种有序键值对的集合,使用枚举类型可以在语义化和结构化之间达成一种平衡。如果我们的代码中到处都是需要文档才能看懂的数字或字符(串)定义的话,那将是一种非常痛苦的事情。老陈最近在学习Java,在对项目重构的时候,就需要枚举类型来优化代码结构,给力的是,Java和.NET等语言(环境)一样都提供了对枚举类型的直接支持package ObjectDemo_java 枚举 键值对

异常检测(三)之线性模型_print(x, correlation=true)-程序员宅基地

文章浏览阅读209次。线性模型即线性方法,本文介绍线性回归与主成成分分析2种方法对数据进行线性建模的两个前提:近似线性相关假设因变量与所有自变量存在线性关系,且与每一个自变量之间都存在线性关系的假设子空间假设子空间假设认为数据是镶嵌在低维子空间中的,线性方法的目的是找到合适的低维子空间使得异常点(o)在其中区别于正常点(n)。数据可视化首先,确定数据集适应的模型,因此需要对数据进行可视化采用的数据集为breast-cancer-unsupervised-ad数据集,下载地址:数据集包下载..._print(x, correlation=true)

收藏 | 语义分割数据集总结-程序员宅基地

文章浏览阅读2.6k次,点赞2次,收藏33次。点击上方“小白学视觉”,选择加"星标"或“置顶”重磅干货,第一时间送达本文转自|视觉算法图片来源于网络语义分割在自然数据集的分割效果不断进步,有研究逐步应用到了遥感领..._公开的sar语义分割数据集

都有哪些原因导致Wi-Fi产品射频电路EVM降低?-程序员宅基地

文章浏览阅读1.3k次,点赞2次,收藏12次。本文只有部分内容,请到一牛网阅读全文: http://www.16rd.com/blog-25466-2979.html 802.11/a/b/g /n WLAN发射机的性能会直接影响产品质量。在当今WLAN产品市场空间拥挤、利润微薄的情况下,提高质量无疑会使产品更具特色并增加其销售量,还能减..._evm调试

Python中json, demjson的使用_damjson-程序员宅基地

文章浏览阅读911次。json中常用的方法在使用的时候先导入json库: import json方法描述json.dumps()将 Python 对象编码成 JSON 字符串json.loads()将已编码的 JSON 字符串解码为 Python 对象json.dump()将Python内置类型序列化为json对象后写入文件json.load()读取文件中json形式的字符串元素转化为Python类型用法json.dumps()import jsonjsonSt_damjson

随便推点

Matlab2019b中配置最小均方误差滤波器(dsp.LMSFilter)详细设置-程序员宅基地

文章浏览阅读6.5k次,点赞3次,收藏35次。主动降噪设计中涉及的最小均方误差算法(LMS)2019b版本中AudioToolbox添加的新功能设置注意:本程序与2016b版本以前不兼容主动降噪设计中最核心的算法莫过于LMS了,实在是太经典了,目前主流的多为LMS算法演化。后面会介绍LMS function的编写,在这里仅介绍如何快速的建立LMS滤波器并进行仿真设计。%% Least mean square filter 自适应滤波器工..._dsp.lmsfilter

51单片机时间戳相关函数_unix时间戳c51单片机-程序员宅基地

文章浏览阅读4.2k次,点赞5次,收藏9次。函数使用了long变量,比较占RAM,单片机要是空间紧张就别用了,会把mcu算糊涂的。/******* timestamp时间戳函数 开始**********/#define SECOND_OF_DAY 86400 //一天多少秒idata uchar DayOfMon[]={31,28,31,30,31,30,31,31,30,31,30,31};/*******************..._unix时间戳c51单片机

C语言之指针篇,快速上手系列-C语言之指针篇(一)-程序员宅基地

文章浏览阅读205次。指针的灵活运用使得c语言更加强大,指针是C语言中十分重要的部分,可以说指针是C语言的灵魂。当然指针不是万能的,但没有指针是万万不能的,有些操作没有指针是办不到的,如动态内存分配。鉴于学习指针的必要性,从现在开始介绍指针方面的知识,本篇主要介绍指针相关概念及指针的定义与应用两方面的内容:指针相关概念1、指针:我们使用的计算机内存为8G,系统为了更好地管理我们的内存,就为内存区的每一个字节都分配一个编..._c语言指针项目练手

软件测试:黑盒测试用例的四种设计方法_黑盒测试用例常用的设计方法-程序员宅基地

文章浏览阅读651次。在对每一输出的同时进行边值分析时,要先确定输出域的全部边值,再设计不同的数据覆盖输出域的边值,这样就能有效地保证输出域的边值被覆盖。对于在测试行业发展的小伙伴们来说应该会很有帮助,有需要的朋友你可以dd我。不能盲目猜想,不能盲目猜测,需要了解系统的薄弱之处和开发人员的盲点,还需要根据以前的缺陷分析报告,分析系统中最容易出现错误的地方,以此作为误判方法的依据。在测试过程中,需要分析每个输出的等价类,在输出域中通常需要确定输出域的所有可能情况,然后对输出的结果进行分类,最后需要设计输入来覆盖所有输出的结果。_黑盒测试用例常用的设计方法

Retinanet网络的安装问题:error: command 'gcc' failed with exit status 1的解决办法_error: command 'gcc' failed with exit status 1重装-程序员宅基地

文章浏览阅读1.4w次。一:安装步骤1.打开Git,克隆retinanet代码到本地:git clone https://github.com/fizyr/keras-retinanet.git2.在当前存储目录下keras-retinanet-master中,执行:pip3 install . --userNOTE:由于自定义安装tensorflow,这个软件包并没有定义对tensorf..._error: command 'gcc' failed with exit status 1重装

[zz]IPv6名称到地址的转换函数getaddrinfo()一些使用心得_python ipv6 全称-程序员宅基地

文章浏览阅读2.3k次。http://www.newsmth.net/bbscon.php?bid=112&id=30469水木社区(展开完整界面) → 网络编程 → 阅读文章NetPRG 版

推荐文章

热门文章

相关标签