JAVA进阶day07JNI(java调用c)A部分

目前接触的JNI有java调用c和c调用java两类。其中java调用c又有隐式和显示两种映射关系。本笔记针对java调用c的显示映射。本着工程实际够用的原则,不够再回头来补充。
JAVA访问c库需要有三个步骤
1:加载C库
2:建立java函数名到c库函数名的映射关系
3:在java程序里调用函数
先贴代码,依照代码来做分析:

public class JNIDemo {static {        /* 1. load */System.loadLibrary("native"); /* libnative.so */}public native void hello();public static void main (String args[]) {JNIDemo d = new JNIDemo();      /* 2. map java hello <-->c c_hello *//* 3. call */d.hello();}
}

分条缕析做笔记:

  • 1:加载C库和在java程序里调用函数
static {        /* 1. load */System.loadLibrary("native"); /* libnative.so */}

代码中在静态代码块中进行加载c库。可以知道我们要加载的是libnative.so库。放到静态代码块中可以保证该加载能够只执行一次。非常棒的处理。

public native void hello();

native修饰的hello表明这个hello方法不是在java中实现的而是在本地语言中实现的。我们在使用hello函数的时候首先要进行声明。这个声明也可以声明为静态的,比如public native static void hello();这时候就没有必要先实现JNIDemo的实例化静态对象了,也就是说我们可以省掉JNIDemo d=new JNI Demo();这个处理,直接使用JNIDemo.hello();当然我们的main函数也是在JNIDemo类中的,可以直接写为hello()。问题又来了,那么同样的道理为什么不是静态的还需要实现JNIDemo对象呢,这个也是在JNIDemo的main函数中啊?这个就没必要较真了,记住就好,又不是让你去创造java语言,你只不过是用这个工具而已。我还发现这个public native static void hello()跟public static native void hello()是一个效果。无碍乎这个static跟native的前后。到此我们对这个加载c库就算是掌握了。

  • 2:建立java函数名到c库函数名的映射关系
    为了将java中的hello跟c语言中的函数对应起来,这部分的处理是c语言中做的。我们姑且猜测,c语言在什么时候去做这个映射对应呢?我提出一个想法,我们在加载SystemloadLibrary的时候会引起c库文件中的一个触发,在那个触发函数里面做了对应。那么我们从文件中的结构应该是包含 触发 实现。我们贴代码分析:

#include <jni.h>  /* /usr/lib/jvm/java-1.7.0-openjdk-amd64/include/ */
#include <stdio.h>#if 0
typedef struct {char *name;          /* Java里调用的函数名 */char *signature;    /* JNI字段描述符, 用来表示Java里调用的函数的参数和返回值类型 */void *fnPtr;          /* C语言实现的本地函数 */
} JNINativeMethod;
#endifvoid c_hello(JNIEnv *env, jobject cls)
{printf("Hello, world!\n");
}static const JNINativeMethod methods[] = {{"hello", "()V", (void *)c_hello},
};/* System.loadLibrary */
JNIEXPORT jint JNICALL
JNI_OnLoad(JavaVM *jvm, void *reserved)
{JNIEnv *env;jclass cls;if ((*jvm)->GetEnv(jvm, (void **)&env, JNI_VERSION_1_4)) {return JNI_ERR; /* JNI version not supported */}cls = (*env)->FindClass(env, "JNIDemo");if (cls == NULL) {return JNI_ERR;}/* 2. map java hello <-->c c_hello */if ((*env)->RegisterNatives(env, cls, methods, 1) < 0)return JNI_ERR;return JNI_VERSION_1_4;
}

我们看到,果不其然跟我们猜测的一模一样。
的确是分为两部分。最顶层那部分实现了c_hello函数,最下半部分实现了一个JNI_OnLoad触发函数。哇咔咔,一切都这么顺滑。那我们开垦这个触发函数吧。
该函数功能比较单一所以写法肯定是固定的。每次使用我们只需要去修改这么几个点就可以了:
JNI_VERSION_1_4 这个代表的是你使用的jni的版本。

    cls = (*env)->FindClass(env, "JNIDemo");

这个作用就是从env找到JNIDemo这个类,方面后面我们去做映射。故而这个JNIDemo我们做移植的时候是需要去修改的。

    if ((*env)->RegisterNatives(env, cls, methods, 1) < 0)

这个地方也就是我们想要的注册了。后面的1代表你要注册的个数。
我们再看看这个注册methods里面的实现。

static const JNINativeMethod methods[] = {{"hello", "()V", (void *)c_hello},
};

这个模式也是固定的
其中“hello”对应的是java中的函数名字。c_hello对应的c文件中的函数名字。前面的void * 是永远不变的。中间的这个类型signature我是打死也不想记住怎么写的。我们可以在写完java文件之后使用javac JNIDemo.java进行编译然后使用
javah -jni JNIDemo
生成一个JNIDemo.h文件

/* DO NOT EDIT THIS FILE - it is machine generated */
#include <jni.h>
/* Header for class JNIDemo */#ifndef _Included_JNIDemo
#define _Included_JNIDemo
#ifdef __cplusplus
extern "C" {
#endif
/** Class:     JNIDemo* Method:    hello* Signature: ()V*/
JNIEXPORT void JNICALL Java_JNIDemo_hello(JNIEnv *, jclass);#ifdef __cplusplus
}
#endif
#endif

我们从里面摘取

/** Class:     JNIDemo* Method:    hello* Signature: ()V*/

就知道这个signature 就是()V.
同时我们知道我们的c文件的hello函数书写形式应该类似与

JNIEXPORT void JNICALL Java_JNIDemo_hello(JNIEnv *, jclass);

实际中我们的c语言函数书写是

void c_hello(JNIEnv *env, jobject cls)

几乎是一样的,其中jobject 跟jclass我觉得应该是java编译器不同导致的,我感性的认为应该是兼容的,都能用。既然我的编译器出来的是jclass那么我的代码就修改为

void c_hello(JNIEnv *env, jclass cls)

写到这里,还剩下两件事,怎样编译c文件为lib*.so以及java同c语言文件如何传递返回数据类型。我们知道java跟c的数据类型上是有区别的。包括开辟的空间大小等。最起码的java中有string类型,然而我们的c语言中就没有。因此,这部分还算是一个不大不小的知识点。按下不表下集再说。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/404005.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

理解线程的挂起,sleep还有阻塞

线程是靠cpu来运行的&#xff0c;cpu要运行一个线程&#xff08;不说别的&#xff09;最起码就是要占用cpu时间&#xff0c;象Windows这样的多任务操作系统&#xff0c;可以允许多个线程同时运行&#xff0c;所谓的同时运行并不是真正的同时运行&#xff0c;而是轮流运行不同的…

9个小窍门让OS X中Finder用起来更顺手

From: http://digi.tech.qq.com/a/20130309/000051.htm 腾讯数码讯&#xff08;编译&#xff1a; 李斯特&#xff09;Finder是OS X系统上用户与文件系统打交道的主要途径之一&#xff0c;它的默认设置是能满足普通用户绝大多数日常需求的。但我们同样可以通过一些小配置来使它…

JAVA进阶day07JNI(java调用c)B部分

本博文要做两件事 1&#xff1a;怎样将c文件编译成lib*.so 2:java同c语言文件如何传递返回数据类型 现在开始搞起&#xff1a; 1&#xff1a;怎样将c文件编译成lib*.so Linux下编译共享库时&#xff0c;必须加上-fPIC参数&#xff0c;否则在链接时会有错误提示。故而我们的…

DataGridView设置行高

RowTemplate.height50; 其他默认

xcode-select: error: tool 'xcodebuild' requires Xcode错误解决方法

From: http://blog.csdn.net/jymn_chen/article/details/21613745 因为机子里有两个Xcode&#xff0c;所以分别重命名了&#xff0c;但是在运行一个MakeFile时却报了以下错误&#xff1a; [plain] view plaincopyxcodebuild -target "GHUnitIOS (Device)" -configu…

Struts2_模块包含 及Action总结

模块包含 做项目时&#xff0c;当Struts配置文件比较多&#xff0c;需要模块化划分或分开成为多个配置文件时&#xff0c;公共配置放在struts.xml,其余有各自的xml文件&#xff0c;后使用<include>标签把其它的配置文件引入到Struts.xml配置文件中就可以了。 默认Action …

JAVA进阶day08泛型

写这篇博文&#xff0c;我并没有特别大的把握。姑且贴代码做分析。遇到什么知识点分析什么知识点吧。 class Person<T> {private T age;public void setAge(T age) {this.age age;}public T getAge() {return this.age;} }public class Generics {public static void ma…

http://www.iteye.com/topic/114392

http://www.iteye.com/topic/114392 http://wiki.jabbercn.org/RFC6120#STARTTLS.E5.8D.8F.E5.95.86 2.如果初始化实体有能力使用SASL 协商, 它必须&#xff08;MUST&#xff09;在初始化流的头信息中包含一个值为"1.0"的属性version。 STARTTLS扩展的XML命名空间是 …

Python模块——_winreg操作注册表

From: http://blog.sina.com.cn/s/blog_4b5039210100gmsb.html 用python操作修改windows注册表&#xff0c;显然要比用C或者C简单。 主要参考资料&#xff1a;官方文档&#xff1a;http://docs.python.org/library/_winreg.html 通过python操作注册表主要有两种方式&#xff…

AS查看Android系统源码

首先要编译安卓源码。我现在使用的是Android5.0源码。平台是tiny4412平台。 一. 编译Android 系统 进入上述解压得到的android-5.0.2&#xff0c;执行以下命令(注意&#xff1a;“.”号后有一个空格)&#xff1a; .setenv再执行“lunch”命令&#xff0c;输入“fulltiny4412…

利用Bdrive打造个人私有云存储解决方案

Bdrive 一款私有云储存软件&#xff0c;可以自己方便的在 Mac/Windows 下架设服务器&#xff0c;并可以通过 PC、Mac、iOS、Android 跨平台使用。以下简单介绍一下利用Bdrive来完成个人私有云存储解决方案。 第一步&#xff0c;搭建Bdrive云存储服务器 先下载Bdrive服务器程序&…

MVC路由中routes.IgnoreRoute({resource}.axd/{*pathInfo}) 到底什么意思!

转自&#xff1a;http://blog.csdn.net/lvjin110/article/details/24638913 参考&#xff08;1&#xff09; http://www.cnblogs.com/flyfish2012/archive/2013/02/01/2889184.html 我们在开发MVC当中&#xff0c;经常在我们的全局类的路由设置&#xff0c;看到这样的代码&…

修改Cocos2d-X-3.2中的setup.py, 使其能用python3

Cocos2d-x的最新版是v3.2&#xff0c;下载地址为&#xff1a;http://cn.cocos2d-x.org/download/ 在运行setup.py时&#xff0c;他会提示你安装python2.7,因为这个版本是他们"well tested"。 但是我电脑上已经安装了python3.3&#xff0c;又不想因为这事而卸载pytho…

函数 devm_kzalloc()

函数 devm_kzalloc() 和kzalloc()一样都是内核内存分配函数&#xff0c;但是devm_kzalloc()是跟设备(device)有关的&#xff0c;当设备(device)被detached或者驱动(driver)卸载(unloaded)时&#xff0c;内存会被自动释放。另外&#xff0c;当内存不在使用时&#xff0c;可以使用…

第四层交换

一&#xff0c;第四层交换简述   第四层交换的一个简单定义是&#xff1a;它是一种功能&#xff0c;它决定 传输不仅仅依据MAC地址(第二层网桥)或源/目标IP地址(第 三层路由),而且依据TCP/UDP(第四层) 应用端口号。第四层 交换功能就象是虚IP&#xff0c;指向物理服务器。它传…

未能加载文件或程序集“Autofac, Version=3.4.0.0,

遇到这个错误的时候&#xff1a;如下图 未能加载文件或程序集“Autofac, Version3.4.0.0, Cultureneutral, PublicKeyToken17863af14b0044da”或它的某一个依赖项。找到的程序集清单定义与程序集引用不匹配。 只要在config加上 <runtime><assemblyBinding xmlns"…

浅谈关于SRAM与DRAM的区别

从名字上看&#xff0c;SRAM与DRAM的区别只在于一个是静态一个是动态。由于SRAM不需要刷新电路就能够保存数据&#xff0c;所以具有静止存取数据的作用。而DRAM则需要不停地刷新电路&#xff0c;否则内部的数据将会消失。而且不停刷新电路的功耗是很高的&#xff0c;在我们的PC…

字符串系列之最长回文子串

2019独角兽企业重金招聘Python工程师标准>>> 问题描述&#xff1a; 给定一个字符串SA1A2...An&#xff0c;要求找出其最长回文子串&#xff08;Longest Palindromic Substring&#xff09;。所谓回文子串就是S的某个子串Ai...Aj为回文。例如&#xff0c;对字符串Sab…

在中断程序里修改全局变量的童鞋注意啦~(C中的volatile作用 转载~)

一个定义为volatile的变量是说这变量可能会被意想不到地改变&#xff0c;这样&#xff0c;编译器就不会去假设这个变量的值了。精确地说就是&#xff0c;优化器在用到这个变量时必须每次都小心地重新读取这个变量的值&#xff0c;而不是使用保存在寄存器里的备份。下面是volati…

设计模式:单例和简单工厂

单例设计模式&#xff1a;只实例化一个类的对象&#xff01; 1 public class Person2 {3 //1&#xff1a;首先定义一个静态变量4 //2&#xff1a;将该类的构造函数私有化5 //3&#xff1a;定义一个静态方法&#xff0c;将该类对象赋给这个静态变量6 …