Java之IO操作总结

所谓IO,也就是Input与Output的缩写。在java中,IO涉及的范围比较大,这里主要讨论针对文件内容的读写

其他知识点将放置后续章节

对于文件内容的操作主要分为两大类

分别是:

  • 字符流
  • 字节流

其中,字符流有两个抽象类:Writer Reader

其对应子类FileWriter和FileReader可实现文件的读写操作

BufferedWriter和BufferedReader能够提供缓冲区功能,用以提高效率

同样,字节流也有两个抽象类:InputStream OutputStream

其对应子类有FileInputStream和FileOutputStream实现文件读写

BufferedInputStream和BufferedOutputStream提供缓冲区功能

概念:

字符流处理的单元为2个字节的Unicode字符,分别操作字符、字符数组或字符串,而字节流处理单元为1个字节, 操作字节和字节数组。所以字符流是由Java虚拟机将字节转化为2个字节的Unicode字符为单位的字符而成的,所以它对多国语言支持性比较好!如果是 音频文件、图片、歌曲,就用字节流好点,如果是关系到中文(文本)的,用字符流好点.

字节流可用于任何类型的对象,包括二进制对象,而字符流只能处理字符或者字符串; 2. 字节流提供了处理任何类型的IO操作的功能,但它不能直接处理Unicode字符,而字符流就可以。

转换:

在从字节流转化为字符流时,实际上就是byte[]转化为String时,
public String(byte bytes[], String charsetName)

有一个关键的参数字符集编码,通常我们都省略了,那系统就用操作系统的lang

而在字符流转化为字节流时,实际上是String转化为byte[]时,

byte[] String.getBytes(String charsetName)
也是一样的道理

参考链接
字节流与字符流的区别 - hintcnuie - ITeye技术网站

俺当初学IO的时候犯了不少迷糊,网上有些代码也无法通过编译,甚至风格都很大不同,所以新手请注意:

1.本文代码较长,不该省略的都没省略,主要是因为作为一个新手需要养成良好的代码编写习惯

2.本文在linux下编译,类似于File.pathSeparator和File.separator这种表示方法是出于跨平台性和健壮性考虑

3.代码中有些操作有多种执行方式,我采用了方式1…方式2…的表述,只需轻轻解开注释便可编译

4.代码中并没有在主方法上抛出异常,而是分别捕捉,造成代码过长,如果仅是测试,或者不想有好的编程习惯,那你就随便抛吧……

5.功能类似的地方就没有重复写注释了,如果新手看不懂下面的代码,那肯定是上面的没有理解清楚

字符流

实例1:字符流的写入


import java.io.File;import java.io.FileWriter;import java.io.IOException;public class Demo {public static void main(String[] args ) {//创建要操作的文件路径和名称//其中,File.separator表示系统相关的分隔符,Linux下为:/  Windows下为:\\String path = File.separator + "home" + File.separator + "siu" + File.separator + "work" + File.separator + "demo.txt";//由于IO操作会抛出异常,因此在try语句块的外部定义FileWriter的引用FileWriter w = null;try {//以path为路径创建一个新的FileWriter对象//如果需要追加数据,而不是覆盖,则使用FileWriter(path,true)构造方法w = new FileWriter(path);//将字符串写入到流中,\r\n表示换行想有好的w.write("Nerxious is a good boy\r\n");//如果想马上看到写入效果,则需要调用w.flush()方法w.flush();} catch (IOException e) {e.printStackTrace();} finally {//如果前面发生异常,那么是无法产生w对象的 //因此要做出判断,以免发生空指针异常if(w != null) {try {//关闭流资源,需要再次捕捉异常w.close();} catch (IOException e) {e.printStackTrace();}}}}}

实例2:字符流的读取


import java.io.File;import java.io.FileReader;import java.io.IOException;public class Demo2 {public static void main(String[] args ) {String path = File.separator + "home" + File.separator + "siu" + File.separator + "work" + File.separator + "demo.txt";FileReader r = null;try {r = new FileReader(path);//方式一:读取单个字符的方式//每读取一次,向下移动一个字符单位int temp1 = r.read();System.out.println((char)temp1);int temp2 = r.read();System.out.println((char)temp2);//方式二:循环读取//read()方法读到文件末尾会返回-1/*while (true) {int temp = r.read();if (temp == -1) {break;}System.out.print((char)temp);}*///方式三:循环读取的简化操作//单个字符读取,当temp不等于-1的时候打印字符/*int temp = 0;while ((temp = r.read()) != -1) {System.out.print((char)temp);}*///方式四:读入到字符数组/*char[] buf = new char[1024];int temp = r.read(buf);//将数组转化为字符串打印,后面参数的意思是//如果字符数组未满,转化成字符串打印后尾部也许会出现其他字符//因此,读取的字符有多少个,就转化多少为字符串System.out.println(new String(buf,0,temp));*///方式五:读入到字符数组的优化//由于有时候文件太大,无法确定需要定义的数组大小//因此一般定义数组长度为1024,采用循环的方式读入/*char[] buf = new char[1024];int temp = 0;while((temp = r.read(buf)) != -1) {System.out.print(new String(buf,0,temp));}*/} catch (IOException e) {e.printStackTrace();} finally {if(r != null) {try {r.close();} catch (IOException e) {e.printStackTrace();}}}}}

实例3:文本文件的复制


import java.io.File;import java.io.FileReader;import java.io.FileWriter;import java.io.IOException;public class Demo {public static void main(String[] args ) {String doc = File.separator + "home" + File.separator + "siu" + File.separator + "work" + File.separator + "demo.txt";String copy = File.separator + "home" + File.separator + "siu" + File.separator + "life" + File.separator + "lrc.txt";FileReader r = null;FileWriter w = null;try {r = new FileReader(doc);w = new FileWriter(copy);//方式一:单个字符写入int temp = 0;while((temp = r.read()) != -1) {w.write(temp);}//方式二:字符数组方式写入/*char[] buf = new char[1024];int temp = 0;while ((temp = r.read(buf)) != -1) {w.write(new String(buf,0,temp));}*/} catch (IOException e) {e.printStackTrace();} finally {//分别判断是否空指针引用,然后关闭流if(r != null) {try {r.close();} catch (IOException e) {e.printStackTrace();}}if(w != null) {try {w.close();} catch (IOException e) {e.printStackTrace();}}}}}

实例4:利用字符流的缓冲区来进行文本文件的复制


import java.io.BufferedReader;import java.io.BufferedWriter;import java.io.File;import java.io.FileReader;import java.io.FileWriter;import java.io.IOException;public class Demo {public static void main(String[] args ) {String doc = File.separator + "home" + File.separator + "siu" + File.separator + "work" + File.separator + "demo.txt";String copy = File.separator + "home" + File.separator + "siu" + File.separator + "life" + File.separator + "lrc.txt";FileReader r = null;FileWriter w = null;//创建缓冲区的引用BufferedReader br = null;BufferedWriter bw = null;try {r = new FileReader(doc);w = new FileWriter(copy);//创建缓冲区对象//将需要提高效率的FileReader和FileWriter对象放入其构造函数内//当然,也可以使用匿名对象的方式 br = new BufferedReader(new FileReader(doc));br = new BufferedReader(r);bw = new BufferedWriter(w);String line = null;//读取行,直到返回null//readLine()方法只返回换行符之前的数据while((line = br.readLine()) != null) {//使用BufferWriter对象的写入方法bw.write(line);//写完文件内容之后换行//newLine()方法依据平台而定//windows下的换行是\r\n//Linux下则是\nbw.newLine();}       } catch (IOException e) {e.printStackTrace();} finally {//此处不再需要捕捉FileReader和FileWriter对象的异常//关闭缓冲区就是关闭缓冲区中的流对象if(br != null) {try {r.close();} catch (IOException e) {e.printStackTrace();}}if(bw != null) {try {bw.close();} catch (IOException e) {e.printStackTrace();}}}}}

字节流

实例5:字节流的写入


import java.io.File;import java.io.FileOutputStream;import java.io.IOException;public class Demo {public static void main(String[] args ) {String path = File.separator + "home" + File.separator + "siu" +File.separator + "work" + File.separator + "demo.txt";FileOutputStream o = null;try {o = new FileOutputStream(path);String str = "Nerxious is a good boy\r\n";byte[] buf = str.getBytes();//也可以直接使用o.write("String".getBytes());//因为字符串就是一个对象,能直接调用方法o.write(buf);} catch (IOException e) {e.printStackTrace();} finally {if(o != null) {try {o.close();} catch (IOException e) {e.printStackTrace();}}}}}

实例6:字节流的读取


import java.io.File;import java.io.FileInputStream;import java.io.IOException;public class Demo {public static void main(String[] args ) {String path = File.separator + "home" + File.separator + "siu" +File.separator + "work" + File.separator + "demo.txt";FileInputStream i = null;try {i = new FileInputStream(path);//方式一:单个字符读取//需要注意的是,此处我用英文文本测试效果良好//但中文就悲剧了,不过下面两个方法效果良好int ch = 0;while((ch=i.read()) != -1){System.out.print((char)ch);}//方式二:数组循环读取/*byte[] buf = new byte[1024];int len = 0;while((len = i.read(buf)) != -1) {System.out.println(new String(buf,0,len));}*///方式三:标准大小的数组读取/*//定一个一个刚好大小的数组//available()方法返回文件的字节数//但是,如果文件过大,内存溢出,那就悲剧了//所以,亲们要慎用!!!上面那个方法就不错byte[] buf = new byte[i.available()];i.read(buf);//因为数组大小刚好,所以转换为字符串时无需在构造函数中设置起始点System.out.println(new String(buf));*/} catch (IOException e) {e.printStackTrace();} finally {if(i != null) {try {i.close();} catch (IOException e) {e.printStackTrace();}}}}}

实例7:二进制文件的复制


import java.io.File;import java.io.FileInputStream;import java.io.FileOutputStream;import java.io.IOException;public class Demo {public static void main(String[] args ) {String bin = File.separator + "home" + File.separator + "siu" +File.separator + "work" + File.separator + "一个人生活.mp3";String copy = File.separator + "home" + File.separator + "siu" +File.separator + "life" + File.separator + "一个人生活.mp3";FileInputStream i = null;FileOutputStream o = null;try {i = new FileInputStream(bin);o = new FileOutputStream(copy);//循环的方式读入写出文件,从而完成复制byte[] buf = new byte[1024];int temp = 0;while((temp = i.read(buf)) != -1) {o.write(buf, 0, temp);}} catch (IOException e) {e.printStackTrace();} finally {if(i != null) {try {i.close();} catch (IOException e) {e.printStackTrace();}}if(o != null) {try {o.close();} catch (IOException e) {e.printStackTrace();}}}}}

实例8:利用字节流的缓冲区进行二进制文件的复制


import java.io.BufferedInputStream;import java.io.BufferedOutputStream;import java.io.File;import java.io.FileInputStream;import java.io.FileOutputStream;import java.io.IOException;public class Demo {public static void main(String[] args ) {String bin = File.separator + "home" + File.separator + "siu" +File.separator + "work" + File.separator + "一个人生活.mp3";String copy = File.separator + "home" + File.separator + "siu" +File.separator + "life" + File.separator + "一个人生活.mp3";FileInputStream i = null;FileOutputStream o = null;BufferedInputStream bi = null;BufferedOutputStream bo = null;try {i = new FileInputStream(bin);o = new FileOutputStream(copy);bi = new BufferedInputStream(i);bo = new BufferedOutputStream(o);byte[] buf = new byte[1024];int temp = 0;while((temp = bi.read(buf)) != -1) {bo.write(buf,0,temp);}} catch (IOException e) {e.printStackTrace();} finally {if(bi != null) {try {i.close();} catch (IOException e) {e.printStackTrace();}}if(bo != null) {try {o.close();} catch (IOException e) {e.printStackTrace();}}}}}

初学者在学会使用字符流和字节流之后未免会产生疑问:什么时候该使用字符流,什么时候又该使用字节流呢?

其实仔细想想就应该知道,所谓字符流,肯定是用于操作类似文本文件或者带有字符文件的场合比较多

而字节流则是操作那些无法直接获取文本信息的二进制文件,比如图片,mp3,视频文件等

说白了在硬盘上都是以字节存储的,只不过字符流在操作文本上面更方便一点而已

此外,为什么要利用缓冲区呢?

我们知道,像迅雷等下载软件都有个缓存的功能,硬盘本身也有缓冲区

试想一下,如果一有数据,不论大小就开始读写,势必会给硬盘造成很大负担,它会感觉很不爽

人不也一样,一顿饭不让你一次吃完,每分钟喂一勺,你怎么想?

因此,采用缓冲区能够在读写大文件的时候有效提高效率

最后解决的一个问题是字节流和字符流的转化,使用的是InputStreamReader和OutputStreamWriter,它们本身属于的是reader和writer字符流,我们之所以会用到这些转化流是因为系统有时候只给我们提供了字节流,为了方便操作,要用到字符流。比如说System.in标准输入流就是字节流。你想从那里得到用户在键盘上的输入,只能是以转换流将它转换为Reader以方便自己的程序读取输入。再比如说Socket里的getInputStream()很明显只给你提供字节流,你要想读取字符,就得给他套个InputStreamReader()用来读取。

package com.zaojiahua.iodemo;import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;public class Test {public static void main(String[] args) throws IOException {//字节流和字符流的相互转化FileInputStream fileInputStream = new FileInputStream("input.txt");//inputSreamReader本来就是reader对象,创建的时候需要传入一个InputStream对象,将字节流转化为字符流BufferedReader reader = new BufferedReader(new InputStreamReader(fileInputStream));//将字符流转化为字节流OutputStreamWriter outputStreamWriter = new OutputStreamWriter(new FileOutputStream("output.txt"));BufferedWriter writer = new BufferedWriter(outputStreamWriter);//实现拷贝文件的操作String buf;while((buf = reader.readLine()) != null){writer.write(buf);writer.newLine();System.out.println(buf);}//关闭流reader.close();writer.close();}
}

参考链接

java中的IO操作总结(一) - Nerxious - 博客园

java中的IO操作|皂荚花

Java之美[从菜鸟到高手演变]之Java中的IO - 智慧演绎,无处不在 - 博客频道 - CSDN.NET

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

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

相关文章

谷歌无人车离奇车祸曝光:人类安全员睡着后,误触关闭了自动驾驶

来源:量子位离奇,真离奇。一场已经被掩盖数月的谷歌无人车(Waymo)离奇车祸,刚刚被The Information曝光。车祸的发生地,就在距离谷歌加州山景城总部不远的高速公路上。那是一个六月的早晨。一辆Waymo无人车正…

强烈推荐:240多个jQuery插件

概述 jQuery 是继 prototype 之后又一个优秀的 Javascript 框架。其宗旨是—写更少的代码,做更多的事情。它是轻量级的 js 库(压缩后只有21k) ,这是其它的 js 库所不及的,它兼容 CSS3,还兼容各种浏览器(IE 6.0, FF 1.5, Safari 2.…

揭秘|超乎想象!未来50年将出现的九大黑科技……

来源:世界科技创新论坛Insititute for the Future是一家专门做预测的研究机构。该机构的首席研究总监Mark Frauenfelder详细描述了人类在未来能够用上,而今天却仍被认为不可能发生的事物。赶快一起来看看这些未来将出现的黑科技吧。1、大脑移植&#xff…

Android手绘效果实现

效果图 原理 大概介绍一下实现原理。首先你得有一张图(废话~),接下来就是把这张图的轮廓提取出来,轮廓提取算法有很多,本人不是搞图像处理的,对图像处理感兴趣的童鞋可以查看相关资料。如果你有好的轮廓提取算法&…

干货|120页精华PPT详解工业机器人本体设计运算及仿真

来源:哈尔滨工业大学摘要:120页精华PPT详解工业机器人本体设计运算及仿真未来智能实验室是人工智能学家与科学院相关机构联合成立的人工智能,互联网和脑科学交叉研究机构。未来智能实验室的主要工作包括:建立AI智能系统智商评测体…

环信SDK集成

利用环信SDK可以实现即时通讯,但在集成的过程中碰到了不少的坑。 注意 选择项目路径,这里以最新版环信demo为例 注意:环信的ChatDemoUI这个demo里边因为研发的同事为了照顾老版本的AndroidStudio使用者,已经用eclipse生成了bui…

协作机器人先驱宣布倒闭!累计融资10.3亿元,贝佐斯投资八轮

来源:量子位一家机器人领域的头部玩家,还不是说倒下就倒下了。上个月,协作机器人的先驱Rethink Robotics刚刚宣布出售第2500个机器人产品,而昨天,就又宣布关门大吉了。CEO Scott Eckert在接受外媒The Robot Report采访…

YOLOv8改进 | 2023注意力篇 | MSDA多尺度空洞注意力(附多位置添加教程)

一、本文介绍 本文给大家带来的改进机制是MSDA(多尺度空洞注意力)发表于今年的中科院一区(算是国内计算机领域的最高期刊了),其全称是"DilateFormer: Multi-Scale Dilated Transformer for Visual Recognition"。MSDA的主要思想是…

贝索斯专访:亚马逊帝国大规模业务转型的秘诀

来源:机器之能摘要:这篇文章简明、清晰地揭示出这家似乎没有边界、无拘束公司成功「转身」背后的逻辑与秘诀。与美国的其它科技业巨头不同,亚马逊并没有一个总的企业园区。在其全球的 57.5 万名雇员中,有 4.5 万名员工和管理人员位…

2018及过去20年诺贝尔化学奖获奖者及其贡献!

来源:科学网摘要:2018及过去20年诺贝尔化学奖获奖者及其贡献!2018年诺贝尔化学获的获得者为美国科学家Frances H. Arnold, George P. Smith和英国科学家George P.Winter,他们利用进化的力量为人类造福。获奖的内容分别是研究酶的定…

RxJava学习入门

RxJava是什么 一个词:异步。 RxJava 在 GitHub 主页上的自我介绍是 “a library for composing asynchronous and event-based programs using observable sequences for the Java VM”(一个在 Java VM 上使用可观测的序列来组成异步的、基于事件的程序…

解读2018年诺贝尔化学奖成果:用进化的力量解决化学问题

来源:新华网摘要:新华社斯德哥尔摩10月3日电 科普:用进化的力量解决化学问题——解读2018年诺贝尔化学奖成果地球的生命经过长期进化最终获得强大的适应力&am…

在win8下安装使用java及在win8下部署java环境变量-图文

为了反编译APK,不得不安装一些Androidfby、apktool1.4.1、dex2jar-0.0.9.9等,甚至连DW也安装了,但是我的电脑是win8X64的,也就是64位的win8系统,这就有点头疼了,出现了: 不是内部或外部命令&…

国际运营商智慧城市探索与实践

来源:中国信息通信研究院CAICT摘要:AT&T、SKT、沃达丰在智慧城市领域探索的经验。2008年11月,IBM提出“智慧地球” 理念引发产业界热议,2010年,该公司进一步提出“智慧城市”愿景作为“智慧地球”在城市运营中的具…

Retrofit学习入门

Retrofit的使用 设置权限与添加依赖 定义请求接口通过创建一个retrofit生成一个接口的实现类(动态代理)调用接口请求数据 设置权限与添加依赖 权限&#xff1a;首先确保在AndroidManifest.xml中请求了网络权限 &#xff1a; <uses-permission android:name"android…

autotools入门笔记(一)

GNU autotools作用&#xff1a;收集系统配置信息并自动生成Makefile文件。 GNU autotools主要包括三个工具&#xff1a;autoconf、automake、libtool&#xff0c;还有很多辅助的工具&#xff0c;包括&#xff1a;autoheader、aclocal、autoscan。 ● autoscan检测源文件生成con…

本田、大众宣布智能路口研究新进展 以安全为重点

编译&#xff1a;网易智能摘要&#xff1a;每年&#xff0c;在十字路口发生的交通事故约占交通事故死亡人数的20%&#xff0c;这就是为什么汽车制造商和供应商都下定决心要创造出智能的十字路口&#xff0c;利用尖端技术来减少&#xff08;或者最好是能够避免&#xff09;在十字…

智能连接:5G、AI和IoT的组合如何改变美洲

来源&#xff1a;199IT互联网数据中心摘要&#xff1a;GSMA发布了新报告“智能连接&#xff1a;5G、AI和IoT的组合如何改变美洲”&#xff0c;强调了该地区如何从这些科技中受益。GSMA Intelligence预测&#xff0c;到2025年全球5G连接数量将达到13亿&#xff0c;覆盖全球40%的…

刚刚,生物学横扫诺贝尔3大奖,两名女性获奖!化学奖授予试管中的“进化论”...

来源&#xff1a;Deeptech深科技北京时间 10 月 3 日下午 5 点 45 分&#xff0c;2018 诺贝尔化学奖揭晓——诺贝尔委员会宣布&#xff0c;将此奖项一半颁给女科学家Frances H. Arnold&#xff0c;另一半则由George P. Smith、 Gregory P. Winter两人共享。此次诺贝尔化学奖表彰…

MACIOS Socket编程

转自 https://github.com/kejinlu/objc-doc/blob/master/Socket%E7%BC%96%E7%A8%8B.md 大纲 一.Socket简介二.BSD Socket编程准备 1.地址2.端口3.网络字节序4.半相关与全相关5.网络编程模型三.socket接口编程示例四.使用select五.使用kqueue六.使用流注:文档中设计涉及的代码也…