Java 文件流操作.
一、概念
在Java中,文件的输入和输出是通过流(Stream)来实现的。一个流,必有源端和目的端,它们可以是计算机内存的某些区域,也可以是磁盘文件,甚至可以是 Internet 上的某个 URL。对于流而言,我们不用关心数据是如何传输的,只需要向源端输入数据,从目的端获取数据即可。
流按照处理数据的单位,可以分为字节流和字符流。字节流的处理单位是字节,通常用来处理二进制文件,例如音乐、图片文件等。而字符流的处理单位是字符,因为Java采用Unicode编码,Java字符流处理的即为Unicode字符,所以在操作汉字、国际化等方面,字符流具有优势。
二、字节流
所有的字节流类都继承自InputStream 和 OutputStream 这两个抽象类,下面列举了5个输入字节流类,输出字节流类和输入字节流类存在对应关系,这个就不一一列举了。
- FileInputStream:把一个文件作为输入源,从本地文件系统中读取数据字节,实现对文件的读取操作。
- ByteArrayInputStream:把内存中的一个缓冲区作为输入源,从内存数组中读取数据字节。
- ObjectInputStream:对以前使用过ObjectOuputStream写入的基本数据和对象进行反序列化,用于恢复那些以前序列化的对象,注意这个对象所属的类必须实现Serializable接口。
- PipeInputStream:实现了管道的概念,从线程通道中读取线程字节。主要在线程中使用,用于两个线程间通信。
- SequenceInputStream:表示其他输入流的逻辑串联。它从输入流的有序集合开始,并从第一个输入流开始读取,直到到达文件末尾,接着从第二个输入流读取,依次类推,直到到达包含的最后一个输入流的文件末尾为止。
- System.in:从用户控制台读取数据字节,在System类中,in是 InputStream 类的静态导入。
public static void main(String[] args) { InputStream in = null; OutputStream out = null; try { //得到输入流 in = new FileInputStream("E:\\test\\a.txt"); //得到输出流 File file = new File("E:\\test\\b.txt"); if (!file.exists()) { file.createNewFile(); } out = new FileOutputStream(file, true); int i;//从输入流读取一定数量的字节,返回 0 到 255 范围内的 int 型字节值 while ((i = in.read()) != -1) { out.write(i); } } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } finally { try { if (in != null) { in.close(); } if (out != null) { out.close(); } } catch (IOException e) { e.printStackTrace(); } } }
三、字符流
所有的字符流类都继承自Reader 和 Writer 这两个抽象类,其中Reader是用于读取字符流的抽象类,Writer是用于写入字符流的抽象类。
Reader 和 Writer 要解决的最主要问题是国际化。原先的 I/O 类库只支持8位的字节流,因此不能很好的处理16位的Unicode字符。Unicode 是国际化的字符集,这样增加了Reader 和 Writer之后,就可以自动在本地字符集和Unicode国际化字符集之间进行转换。
- FileReader:与FileInputStream对应,从文件系统中读取字符序列。
- CharArrayReader:与ByteArrayInputStream 对应,从字符数组中读取数据。
- PipedReader:与PipedInputStream 对应,从线程管道中读取字符序列。
- StringReader:从字符串中读取字符序列。
/** * 由于是字符,存在编码不一致导致乱码的问题 * @param args */ public static void main(String[] args) { Reader reader = null; Writer writer = null; try { //得到输入流 reader = new FileReader("E:\\test\\a.txt"); //得到输出流 writer = new FileWriter("E:\\test\\c.txt", true); char[] chars = new char[50]; int i; while ((i = reader.read(chars)) != -1) { writer.write(chars, 0, i); writer.flush(); } } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } finally { try { if (reader != null) { reader.close(); } if (writer != null) { writer.close(); } } catch (IOException e) { e.printStackTrace(); } } }
四、缓冲流
前面介绍的字节流、字符流都是无缓冲的输入、输出流,这就意味着,每一次的读、写操作都会交给操作系统来处理。这样的做法可能会对系统的性能造成很大的影响,因为每一次操作都可能引起磁盘硬件的读、写或网络的访问。因此,对于字节流和字符流,一般不直接使用。
缓存流是一种装饰器类,目的是让原字节流、字符流 新增缓冲的功能。以字符缓冲流为例进行说明,字符缓冲流从字符流中读取、写入字符,不立刻要求系统进行处理,而是缓冲部分字符,从而实现按规定字符数、按行等方式高效ed读取或写入。
字节缓冲流:
public static void main(String[] args) { BufferedInputStream in = null; BufferedOutputStream out = null; try { in = new BufferedInputStream(new FileInputStream("E:\\test\\a.txt")); out = new BufferedOutputStream(new FileOutputStream("E:\\test\\e.txt", true)); byte[] b = new byte[1024]; int i; while ((i = in.read(b)) != -1) { out.write(b, 0, i); out.flush();//手动刷新该流的缓冲,立即将他们写入预期目标 } } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } finally { try { if (in != null) { in.close(); } if (out != null) { out.close(); } } catch (IOException e) { e.printStackTrace(); } } }
字符缓冲流:
public static void main(String[] args) { BufferedReader bufferedReader = null; BufferedWriter bufferedWriter = null; try { //设置文件编码,解决文件乱码问题 //将字节流转换为字符流,实际上使用了一种设计模式——适配器模式 InputStreamReader isr = new InputStreamReader(new FileInputStream("E:\\test\\a.txt"), "GBK"); bufferedReader = new BufferedReader(isr); bufferedWriter = new BufferedWriter(new FileWriter("E:\\test\\d.txt")); String s; while ((s = bufferedReader.readLine()) != null) { bufferedWriter.write(s); bufferedWriter.newLine();//按行读取,写入一个分行符,否则所有内容都在一行显示 } } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } finally { try { if (bufferedReader != null) { bufferedReader.close(); } if (bufferedWriter != null) { bufferedWriter.close(); } } catch (IOException e) { e.printStackTrace(); } } }
低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
从零开始学 Web 之 JavaScript(一)JavaScript概述
大家好,这里是「 Daotin的梦呓 」从零开始学 Web 系列教程。此文首发于「 Daotin的梦呓 」公众号,欢迎大家订阅关注。在这里我会从 Web 前端零基础开始,一步步学习 Web 相关的知识点,期间也会分享一些好玩的项目。现在就让我们一起进入 Web 前端学习的冒险之旅吧! 一、JavaScript 概述 1、JavaScript简介 JavaScript历史 要了解JavaScript,我们首先要回顾一下JavaScript的诞生。在上个世纪的1995年,当时的网景公司正凭借其Navigator浏览器成为Web时代开启时最著名的第一代互联网公司。由于网景公司希望能在静态HTML页面上添加一些动态效果,于是叫Brendan Eich这哥们在两周之内设计出了JavaScript语言。你没看错,这哥们只用了10天时间。 为什么起名叫JavaScript?原因是当时Java语言非常红火,所以网景公司希望借Java的名气来推广,但事实上JavaScript除了语法上有点像Java,其他部分基本上没啥关系。 ECMAScript 因为网景开发了JavaScript,一年后微软又模仿Ja...
- 下一篇
Docker 常用命令总结
列出镜像列表 docker images 获取新的镜像 docker pull centos:7.0 查找镜像 docker search httpd 更新镜像 docker commit -m="提交的信息" -a="作者" e218edb10161 更改后的镜像名 删除镜像 docker rmi 49c614fbbea8 Error response from daemon: conflict: unable to delete 49c614fbbea8 (must be forced) - image is referenced in multiple repositories # 这个报错说明它被其他的镜像依赖,所以需要先删除依赖的image [root@FantJ ~]# docker rmi 49c614fbbea8 Error response from daemon: conflict: unable to delete 49c614fbbea8 (must be forced) - image is being used by stopped container ff...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- SpringBoot2配置默认Tomcat设置,开启更多高级功能
- CentOS7编译安装Gcc9.2.0,解决mysql等软件编译问题
- CentOS7安装Docker,走上虚拟化容器引擎之路
- SpringBoot2更换Tomcat为Jetty,小型站点的福音
- Springboot2将连接池hikari替换为druid,体验最强大的数据库连接池
- Docker快速安装Oracle11G,搭建oracle11g学习环境
- Docker使用Oracle官方镜像安装(12C,18C,19C)
- SpringBoot2全家桶,快速入门学习开发网站教程
- CentOS8编译安装MySQL8.0.19
- Jdk安装(Linux,MacOS,Windows),包含三大操作系统的最全安装