JNI技术2---本地代码反调用java类方法过程详解

简介: 记得在前面的JNI中我介绍了  在 java中调用本地代码的例子 ,这次 我就反过来   。 (如果初次接触JNI请看 JNI技术-1上篇文章) 首先我们在利用javah工具 生成 java类所对应的 C++头文件的时候 ,我们观察 java本地方法相关的定义 .

记得在前面的JNI中我介绍了  在 java中调用本地代码的例子 ,这次 我就反过来   。 (如果初次接触JNI请看 JNI技术-1上篇文章)

首先我们在利用javah工具 生成 java类所对应的 C++头文件的时候 ,我们观察 java本地方法相关的定义 ..

JNIEXPORT void JNICALL Java_me_Native_Reflect_show (JNIEnv *, jobject); 

  //看这里2个参数 ,JNIEnv 类型的指针 ,和jobject类型的对象     他们分别代表了 JRE (java 运行时环境)  和 调用这个本地方法的 类的对象 或者 类的Class对象  。

JNIEnv其实是定义的一个结构体  里面包含了 许许多多的函数  比如 GetMethodID() 获取方法的 jmethodID对象     

                                                                                                                          GetStaticMethodID()获取静态方法的jmethodID对象 

                                                                                                                          CallStaticVoidMethod()调用静态 无返回值的方法等等  许许多多类似应用在JNI.H头文件中大家自己去查看   就清晰可见了  ,关于这2个对像是如何传递进来的呢 ?在调用本地方法的同时由JNI传递  比如我们担心

 

 因为 在JNI中 从 java到 C++有引用类型 和基本数据类型的过度,  也就是  在 java中没中数据类型 都有 一个与之对应的 C++类型 

  例如我们的 java中的 int 类型 在 C+中对于 jint类型 ,一般情况下都是在java数据类型 和引用类型前面加上一个 j开头 表示 是java类型的对应、

我们在 jni.h头文件中 找到如下定义

class _jobject {};
class _jclass : public _jobject {};
class _jthrowable : public _jobject {};
class _jstring : public _jobject {};
class _jarray : public _jobject {};
class _jbooleanArray : public _jarray {};
class _jbyteArray : public _jarray {};
class _jcharArray : public _jarray {};
class _jshortArray : public _jarray {};
class _jintArray : public _jarray {};
class _jlongArray : public _jarray {};
class _jfloatArray : public _jarray {};
class _jdoubleArray : public _jarray {};
class _jobjectArray : public _jarray {};
等等 的类型 还有很多 ,我们发现他们很有层次感  

所有对象 直接或者间接的继承了 _jobject类 ,这也是为了 和 java中 以object为基类的类层次相对应,

 

下面是C++和 java互相调用的2段代码 

在C++中饭调用java     在上次的操作之后修改 JNI方法的实现部分

#include <iostream>
#include "me_Native_Reflect.h"
using namespace std ;
JNIEXPORT void JNICALL Java_me_Native_Reflect_show(JNIEnv * env, jobject obj)//java本地方法的实现
{
     cout<<"java调用本地C++成功!"<<endl ;
  jclass cls =env->GetObjectClass(obj) ;  //获得JAVA对象在C++中的对应对象  jclass
  jmethodID  id=env->GetStaticMethodID(cls,"showStatic","()V") ;  //通过 JNIEnv类 也就是java环境获得  静态方法的 jmethodID          

//第三个参数 是方法的 返回类型参数签名  可以用  javap工具来查看 一个类的所有属性和方法  以及他们的签名  。。。。。
  env->CallStaticVoidMethod(cls,id) ;  //INVOKE  STATIC  METHOD   执行java静态方法 
}

 

//在java中调用C++   这是java代码

package me.Native;
public  class  Reflect
{
 public  native void  show()  ; 
 public  static void showStatic()
 {
  System.out.println("c++调用java静态方法成功!") ;
 }
 public  static  void main(String[]args)
 {  
  System.loadLibrary("JNIReflect") ;
  Reflect r=new Reflect() ;
  r.show() ;
 }

}

目录
相关文章
|
1月前
|
Java
Java语言实现字母大小写转换的方法
Java提供了多种灵活的方法来处理字符串中的字母大小写转换。根据具体需求,可以选择适合的方法来实现。在大多数情况下,使用 String类或 Character类的方法已经足够。但是,在需要更复杂的逻辑或处理非常规字符集时,可以通过字符流或手动遍历字符串来实现更精细的控制。
223 18
|
1月前
|
Java 编译器 Go
【Java】(5)方法的概念、方法的调用、方法重载、构造方法的创建
Java方法是语句的集合,它们在一起执行一个功能。方法是解决一类问题的步骤的有序组合方法包含于类或对象中方法在程序中被创建,在其他地方被引用方法的优点使程序变得更简短而清晰。有利于程序维护。可以提高程序开发的效率。提高了代码的重用性。方法的名字的第一个单词应以小写字母作为开头,后面的单词则用大写字母开头写,不使用连接符。例如:addPerson。这种就属于驼峰写法下划线可能出现在 JUnit 测试方法名称中用以分隔名称的逻辑组件。
196 5
|
1月前
|
编解码 Java 开发者
Java String类的关键方法总结
以上总结了Java `String` 类最常见和重要功能性方法。每种操作都对应着日常编程任务,并且理解每种操作如何影响及处理 `Strings` 对于任何使用 Java 的开发者来说都至关重要。
263 5
|
2月前
|
算法 安全 Java
除了类,Java中的接口和方法也可以使用泛型吗?
除了类,Java中的接口和方法也可以使用泛型吗?
134 11
|
Java
使用Java代码打印log日志
使用Java代码打印log日志
437 1
|
Java BI API
在Java代码中打日志需要注意什么?
日志是什么?日志是你在代码运行时打印出来的一些数据和记录,是快速排查问题的好帮手,是撕逼和甩锅的利器!
836 0
|
缓存 Java 网络架构
别在 Java 代码里乱打日志了,这才是正确的打日志姿势!
别在 Java 代码里乱打日志了,这才是正确的打日志姿势!
231 0
|
缓存 架构师 搜索推荐
别在 Java 代码里乱打日志了,这才是正确的日志打印姿势!
使用门面模式的日志框架,有利于维护和各个类的日志处理方式统一。
|
Java BI Apache
在Java代码中打日志需要注意什么?
云栖号资讯:【点击查看更多行业资讯】在这里您可以找到不同行业的第一手的上云资讯,还在等什么,快来! 为什么要打日志? 日志是什么?日志是你在代码运行时打印出来的一些数据和记录,是快速排查问题的好帮手! 做一件事情之前,先思考为什么。
在Java代码中打日志需要注意什么?
|
Java Android开发 C语言
02_JNI中Java代码调用C代码,Android中使用log库打印日志,javah命令的使用,Android.mk文件的编写,交叉编译
 1  编写以下案例(下面的三个按钮都调用了底层的C语言): 项目案例的代码结构如下: 2 编写DataProvider的代码: package com.example.ndkpassdata;   public class DataProvider {         /**      * 计算x和y的加法  apktools      *
1457 0