深入分析Java I/O 工作机制
前言 : I/O 问题是Web 应用中所面临的主要问题之一。而且是任何编程语言都无法回避的问题,是整个人机交互的核心。
java 的I/O类操作在java.io 包下,将近80个子类, 大概可以分成四个部分,
- 基于字节操作的I/O接口 :InputStream 和OutputStream
- 基于字符操作的I/O接口:Writer 和 Reader
- 基于磁盘操作的I/O接口:File
- 基于网络操作的I/O接口:Socket
1和2 主要是传输数据的格式,后面两个则是传输数据的方式。 要么数据格式影响I/O操作,要么传输方式影响I/O操作,核心问题大致可以归为这两个 。
基于字节的I/O操作接口
InputStream类层次结构
OutputStream类层次结构
操作数据的方式是可以组合使用的 例如
OutputStream out = new BufferedOutputStream (new ObjectOutputStream (new FileOutputStream ("filename")));
必须要指定数据流最终写入的地方,要么写入到磁盘,要么写到网络中,
基于字符的I/O操作接口
不论是磁盘还是网络传输,最小的存储单元都是字节,而不是字符,所以,I/O操作的都是字节而不是字符,但是为什么会有操作字符的I/O 接口?
我们在程序中通常操作的数据都是字符形式(Java中char占用2字节,C/C++中 char占用1字节),为了操作方便当然会提供一个字符接口,字符到字节必须要经过编码的转换,编码转换又十分耗时,还会出现乱码问题
Writer 类层次结构
Writer 类提供了一个抽象方法write(char cbuf[ ],int off,in len)
Reader 类层次结构
读字符的操作接口是 int read (char cbuf[ ],int off,in len),返回读到的n个字节数,不管是读还是写,他们都只定义了读取或写入的数据字符的方式,即怎么写,怎么读,并没有规定数据要读或写到哪里
字节与字符的转换接口
有时数据持久化和网络传输是以字节进行的,所有需要字节和字符之间的相互转换
InputStreamReader 类是从字节到字符的转化桥梁,从InputStream 到Reader 的过程要指定编码字符集,否则会采用操作系统默认的字符集,就会出现乱码的问题,StreamDecoder就是完成从字节到字符的编码的实现类
try{
StringBuffer str = new StringBuffer();
char [] buf = new char[1024]
FileReader f = new FileReader("filename")
while(f.read(buf)>0){
str.append(buf);
}
str.tostring(); }catch (IOException e){ }
FileReader 就是按照上面的工作方式来读取文件的,FileReader 类继承了InputStreamReader类,实际上是读取文件流,然后通过StreamDecoder 解码成char,这里的解码字符集是默认字符集。
磁盘I/O工作机制
1, 标准访问文件的方式
当应用程序调用read() 接口时候,操作系统检查在内核的高速缓存中有没有需要的数据,如果已经缓存了那么就直接从缓存中取出返回,如果没有,则从磁盘中读取,然后缓存在操作系统的缓存中。
写入的方式是,用户的应用程序调用write() 接口将数据从用户地址空间复制到内核地址空间的缓存中,这时对用户程序来说,写操作就已经完成了,至于什么时候写入到磁盘中去,由操作系统决定,除非显示的调用了sync同步命令。
标准文件访问方式
2, 直接I/O的方式
即应用程序直接访问磁盘数据而不经过操作系统内核数据缓冲区,这样减少了一次从内核缓冲区到用户程序缓存的复制, 这种访问方式通常在对数据的缓存管理由应用程序实现的数据库管理系统中,在数据库管理系统中,系统明确的知道应该缓存哪些数据,应该失效哪些数据,还可以对一些热点数据做预加载,提前将数据加载到内存,可以加速数据访问效率,在这些情况下,如果由操作系统进行缓存就很难做到,操作系统不知道哪些是热点数据,哪些数据只会访问一次就不会再访问,操作系统只是简单的缓存最近一次从磁盘读取的数据,
但是直接I/O的话也会有负面影响: 如果访问的数据不在应用程序缓存中,那么每次数据都要直接从磁盘进行预加载,这种直接加载会十分缓慢,通常直接I/O 与异步I/O结合使用会有比较好的性能。
直接I/O 的方式
3, 同步访问文件的方式
同步访问文件的方式比较容易理解,就是数据读取和写入都是同步操作的,与标准文件访问方式不同的是,只有当数据被成功写到磁盘时才返回给应用程序成功的标志, 这种访问方式的性能比较差,只有在对数据安全性较高的场景中才会使用,而且通常这种操作方式的硬件都是定制的。
同步访问文件的方式
4, 异步访问文件的方式
异步访问文件的方式是当访问数据的线程发出请求之后,线程会接着去处理其他事,而不是阻塞等待,当请求的数据返回之后,继续处理下面的操作, 这种访问文件的方式可以明显的提高应用程序的效率,但是不会改变访问文件的效率
异步访问文件的方式
5, 内存映射的方式
内存映射是指操作系统将内存中的某一块区域与磁盘中的文件关联起来,当要访问内存中的一段数据的时候,转换为访问文件的某一段数据,这种方式的目的同样是减少数据从内核空间缓存到用户空间缓存的数据复制操作,因为这两个数据空间的数据是共享的。
内存映射方式
深入分析Java I/O 工作机制的更多相关文章
- 2 深入分析 Java IO的工作机制(一)
大部分Web应用系统的瓶颈都是I/O瓶颈 2.1 Java的I/O类库的基本架构 Java的I/O操作类在包java.io下,大概有将近80个类,这些类大概可以分成如下4组. 基于字节操作的I/O接口 ...
- 2 深入分析 Java IO的工作机制(二)
2.5 I/O调优 下面总结一些磁盘I/O和网络I/O的常用优化技巧. 2.5.1 磁盘I/O优化 1. 性能检测 应用程序通常都需要访问磁盘来读取数据,而磁盘I/O通常都很耗时,要判断I/O是否是一 ...
- java I/O工作机制
java I/O 的基本架构: 1:基于字节操作的I/O接口 InputStream OutputStream 2:基于字符操作的I/O接口 Writer 和Reader 3:基于磁盘操作的I/O接口 ...
- Java I/O 工作机制(一) —— Java 的 I/O 类库的基本架构
Java 的 I/O 类库的基本架构 Java 的 I/O 操作类在包 java.io 下,有将近 80 个类. 按数据格式分类: 面向字节(Byte)操作的 I/O 接口:InputStream 和 ...
- Java I/O 工作机制(二) —— Java 的 I/O 的交互方式分析
简介: BIO:同步阻塞式IO,服务器实现模式为一个连接一个线程,即客户端有连接请求时服务器端就需要启动一个线程进行处理,如果这个连接不做任何事情会造成不必要的线程开销,当然可以通过线程池机制改善. ...
- Java Web ClassLoader工作机制
一.ClassLoader的作用: 1.类加载机制:父优先的等级加载机制 2.类加载过程 3.将Class字节码重新解析成JVM统一要求的对象格式 二.ClassLoader常用方法 1.define ...
- Java Socket 的工作机制
转载,请加上原文链接: 目录 socket 对象的创建时间 socket 通信可能会造成死锁 socket 对象的创建时间 这里需要一点TCP的知识, TCP状态分析请看 --> TCP转态转换 ...
- 深入分析 Java I/O 的工作机制--转载
Java 的 I/O 类库的基本架构 I/O 问题是任何编程语言都无法回避的问题,可以说 I/O 问题是整个人机交互的核心问题,因为 I/O 是机器获取和交换信息的主要渠道.在当今这个数据大爆炸时代, ...
- 深入分析 Java I/O 的工作机制
I/O 问题可以说是当今互联网 Web 应用中所面临的主要问题之一,因为当前在这个海量数据时代,数据在网络中随处流动.这个流动的过程中都涉及到 I/O 问题,可以说大部分 Web 应用系统的瓶颈都是 ...
随机推荐
- wireshark使用方法
抓取报文: 下载和安装好Wireshark之后,启动Wireshark并且在接口列表中选择接口名,然后开始在此接口上抓包.例如,如果想要在无线网络上抓取流量,点击无线接口.点击Capture Opti ...
- javascript项目实战之原生js模拟淘宝购物车
通过JavaScript实现类似与淘宝的购物车效果,包括商品的单选.全选.删除.修改数量.价格计算.数目计算.预览等功能的实现.实现的效果图: 相应的代码: shoppingCart.html < ...
- 16.app后端如何保证通讯安全--url签名
app和后端的通讯过程中,api请求有可能被别人截取或不小心泄露.那么,怎么保证api请求的安全呢?在这篇文章中,介绍一种常见的保证api请求安全的做法--url签名. 1. url签名详解 在前一篇 ...
- python使用itchat库实现微信机器人
itchat是一个开源的微信个人号接口,可以使用该库进行微信网页版中的所有操作,比如:所有好友.添加好友.拉好友群聊.微信机器人等等.详细用户请看文档介绍,在这里. 本文主要使用该库完成一个能够处理微 ...
- Axure使用——创建折叠菜单
1.先添加动态面板 2.往动态面板中添加矩形 3.接着先隐藏下面的矩形(也就是你要折叠起来的内容) 4.一定要注意: 5.添加动态面板的状态 6.把之前做的那个矩形全部复制到state1中 7.把之前 ...
- Immutable(不可变)集合
Immutable(不可变)集合 不可变集合,顾名思义就是说集合是不可被修改的.集合的数据项是在创建的时候提供,并且在整个生命周期中都不可改变. 为什么要用immutable对象?immutable对 ...
- bzoj3236 作业 莫队+树状数组
莫队+树状数组 #include<cstdio> #include<cstring> #include<iostream> #include<algorith ...
- BZOJ_4325_NOIP2015 斗地主_DFS
BZOJ_4325_NOIP2015 斗地主_DFS Description 牛牛最近迷上了一种叫斗地主的扑克游戏.斗地主是一种使用黑桃.红心.梅花.方片的A到K加上大小王的共54张牌来进行的扑克牌游 ...
- BZOJ_2001_[BeiJing2006]狼抓兔子_最小割转对偶图
BZOJ_2001_[BeiJing2006]狼抓兔子 题意:http://www.lydsy.com/JudgeOnline/problem.php?id=1001 分析:思路同NOI2010海拔. ...
- java集合框架之Collections
参考http://how2j.cn/k/collection/collection-collections/369.html Collections是一个类,容器的工具类,就如同Arrays是数组的工 ...