JNI 方法注册与签名+BufferedReader使用readLine问题
最近了解了关于JavaJNI接口的一些关于方法注册与签名相关的知识,在此进行一下总结。
使用JNI接口时,我们首先需要把Java方法声明为native:
- public native void f();
然后编写对应的C/C++代码,并编译成为动态链接库(.dll或.so),在调用Java方法前载入动态链接库即可调用:
- static {
- System.loadLibrary("native-lib");
- }
那么,Java文件中的native方法是如何与native文件中的方法一一对应的呢?
在此有两种方法:静态注册与动态注册,下面将一一介绍:
静态注册
- javah -jni 包名.类名
自动生成对应的c层头文件
- package com.app.superxlcr.jnitest;
- /**
- * Created by superxlcr on 2017/5/25.
- */
- public class NativeTest {
- public native void f();
- public native int f(int a, double b);
- public native void f(Object a, String b);
- public native void g();
- }
native层:
- /* DO NOT EDIT THIS FILE - it is machine generated */
- #include <jni.h>
- /* Header for class com_app_superxlcr_jnitest_NativeTest */
- #ifndef _Included_com_app_superxlcr_jnitest_NativeTest
- #define _Included_com_app_superxlcr_jnitest_NativeTest
- #ifdef __cplusplus
- extern "C" {
- #endif
- /*
- * Class: com_app_superxlcr_jnitest_NativeTest
- * Method: f
- * Signature: ()V
- */
- JNIEXPORT void JNICALL Java_com_app_superxlcr_jnitest_NativeTest_f__
- (JNIEnv *, jobject);
- /*
- * Class: com_app_superxlcr_jnitest_NativeTest
- * Method: f
- * Signature: (ID)I
- */
- JNIEXPORT jint JNICALL Java_com_app_superxlcr_jnitest_NativeTest_f__ID
- (JNIEnv *, jobject, jint, jdouble);
- /*
- * Class: com_app_superxlcr_jnitest_NativeTest
- * Method: f
- * Signature: (Ljava/lang/Object;Ljava/lang/String;)V
- */
- JNIEXPORT void JNICALL Java_com_app_superxlcr_jnitest_NativeTest_f__Ljava_lang_Object_2Ljava_lang_String_2
- (JNIEnv *, jobject, jobject, jstring);
- /*
- * Class: com_app_superxlcr_jnitest_NativeTest
- * Method: g
- * Signature: ()V
- */
- JNIEXPORT void JNICALL Java_com_app_superxlcr_jnitest_NativeTest_g
- (JNIEnv *, jobject);
- #ifdef __cplusplus
- }
- #endif
- #endif
我们可以看到,对于拥有重载的f 方法,其native方法名称后都带有参数,而没有重载的g 方法则没带有
动态注册
- typedef struct {
- // Java层native方法名称
- const char* name;
- // 方法签名
- const char* signature;
- // native层方法指针
- void* fnPtr;
- } JNINativeMethod;
然后重写JNI_OnLoad方法(该方法会在Java层通过System.loadLibrary加载完动态链接库后被调用),我们在其中进行动态注册工作:
- JNIEXPORT jint JNICALL
- JNI_OnLoad(JavaVM* vm, void* reserved) {
- JNIEnv *env = NULL;
- jint result = -1;
- // 获取JNI env变量
- if (vm->GetEnv((void**) &env, JNI_VERSION_1_4) != JNI_OK) {
- // 失败返回-1
- return result;
- }
- // 获取native方法所在类
- const char* className = "com/app/superxlcr/jnitest/MainActivity";
- jclass clazz = env->FindClass(className);
- if (clazz == NULL) {
- return result;
- }
- // 动态注册native方法
- if (env->RegisterNatives(clazz, methods, 1) < 0) {
- return result;
- }
- // 返回成功
- result = JNI_VERSION_1_4;
- return result;
- }
动态注册的大致步骤如下:
- 通过vm(Java虚拟机)参数获取JNIEnv变量
- 通过FindClass方法找到对应的Java类
- 通过RegisterNatives方法,传入JNINativeMethod数组,注册native函数
方法签名
- (参数类型标识1参数类型标识2...参数类型标识n)返回值类型标识
类型标识对应关系如下:
| 类型标识 | Java类型 |
| Z | boolean |
| B | byte |
| C | char |
| S | short |
| I | int |
| J | long |
| F | float |
| D | double |
| L包名/类名; | 各种引用类型 |
| V | void |
另外,当Java类型为数组时,在标识前会有“[”符号,例如:String[] 类型标识为 [Ljava/lang/String;
- // Signature: ()V
- public native void f();
- // Signature: (ID)I
- public native int f(int a, double b);
- // Signature: (Ljava/lang/Object;Ljava/lang/String;)V
- public native void f(Object a, String b);
- // Signature: ()V
- public native void g();
- BufferedReader使用readLine问题
有时我们在使用BufferedReader时候会发现使用readLine函数迟迟没有任何返回,这是因为BufferedReader和BufferedWriter是基于行进行操作的,因此我们使用BufferedWriter的时候使用newLine函数即可,具体代码如下:
- BufferedWriter writer = new BufferedWriter(new OutputStreamWriter(out));
- writer.write(str);
- writer.newLine();
- writer.flush();
- BufferedReader reader = new BufferedReader(new InputStreamReader(in));
- str = reader.readLine();
JNI 方法注册与签名+BufferedReader使用readLine问题的更多相关文章
- Dalvik虚拟机JNI方法的注册过程分析
文章转载至CSDN社区罗升阳的安卓之旅,原文地址:http://blog.csdn.net/luoshengyang/article/details/8923483 在前面一文中,我们分析了Dalvi ...
- JNI 函数注册与管理
class<--> 一一对应so-->method 每个so对应于一个类对象 类中的每个native方法对应 于so中的一个native的function,对应关系涉及 {c ...
- Android Studio NDK JNI动态注册本地方法
概述 可能大家觉得javah生成的函数名又臭又长,不太好看.这里可以提供另外一种方法来动态注册c++函数,让其根Java中的native方法关联起来. 实现 这里通过JNIEnv的Resisterna ...
- Java Native Interface 五 JNI里的多线程与JNI方法的注册
本文是<The Java Native Interface Programmer's Guide and Specification>读书笔记 JNI里的多线程 在本地方法里写有关多线程的 ...
- Android有关JNI 学习(两)为JNI方法名称,数据类型和方法签名的一些知识
我们知道,使用javah产生c/c++当在头文件,将java定义 native 功能,以产生相应jni层功能,如下面: /* * Class: com_lms_jni_JniTest * Method ...
- JNI动态注册native方法及JNI数据使用
前言 或许你知道了jni的简单调用,其实不算什么百度谷歌一大把,虽然这些jni绝大多数情况下都不会让我们安卓工程师来弄,毕竟还是有点难,但是我们还是得打破砂锅知道为什么这样干吧,至少也让我们知道调用流 ...
- Java中使用BufferedReader的readLine()方法和read()方法来读取文件内容
目标:读文件 编程时,有很多时候需要读取本地文件,下面介绍一下读取方式: 读单行文件 package com; import java.io.*; import java.util.ArrayList ...
- JNI的第2种写法:本地方法注册
声明:迁移自本人CSDN博客https://blog.csdn.net/u013365635 孔乙己说,茴香豆的茴有四种写法,今天谈谈JNI的第2种写法:本地方法注册. 这种写法的好处是不需要使用ja ...
- 技术转载:Jni学习四:如何编写jni方法
转载:http://blog.chinaunix.net/u1/38994/showart_1099528.html 一.概述: 在这篇文章中将会简单介绍如何编制一些简单的JNI 方法.我们都知道JN ...
随机推荐
- Python内置函数(21)——tuple
英文文档: The constructor builds a tuple whose items are the same and in the same order as iterable's it ...
- python random 模块的用法
Python中的random模块用于生成随机数.下面介绍一下random模块中最常用的几个函数. random.random random.random()用于生成一个0到1的随机符点数: 0 < ...
- ORM “杀器”之 JOOQ
ORM “杀器”之 JOOQ IN 后端编程,JAVA,敏捷开发,数据库 JOOQ是啥? JOOQ 是基于Java访问关系型数据库的工具包,轻量,简单,并且足够灵活,可以轻松的使用Java面向对象语法 ...
- python入门(12)dict
python入门(12)dict Python内置了字典:dict的支持,dict全称dictionary,在其他语言中也称为map,使用键-值(key-value)存储,具有极快的查找速度. 举个例 ...
- leetcode算法: Keyboard Row
Given a List of words, return the words that can be typed using letters of alphabet on only one row' ...
- SpringBoot(五):@ConfigurationProperties配置参数绑定
在springmvc或其他ssh框架中如果我们要实现一个配置参数的加载,需要使用代码实现读取properties文件等操作,或者需要使用其他属性@value(name="username&q ...
- hdu1005 Number Sequence---找循环节
题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=1005题目大意: f(1) = 1, f(2) = 1, f(n) = (A * f(n - 1) + ...
- 浅析开源数据库MySQL架构
数据库是所有应用系统的核心,故保证数据库稳定.高效.安全地运行是所有企业日常工作的重中之重.数据库系统一旦出现问题无法提供服务,有可能导致整个系统都无法继续工作.所以,一个成功的数据库架构在高可用设计 ...
- MySQL中的数值函数
加减乘除(+.-.*./)均可用于数值计算. SELECT (1 + 1) / (1 * 2.2 - 3) 执行算术函数 单参数数值函数举例: acos(x), asin(x), atan(x), c ...
- python3全栈开发-异常处理
一. 什么是异常 异常就是程序运行时发生错误的信号(在程序出现错误时,则会产生一个异常,若程序没有处理它,则会抛出该异常,程序的运行也随之终止),在python中,错误触发的异常如下 而错误分成两种 ...