Java—IO流详解(一)

2024-04-27 12:58
文章标签 java io 详解

本文主要是介绍Java—IO流详解(一),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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

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

分别是:

  字符流
  字节流

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

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

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


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

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

BufferedInputStream和BufferedOutputStream提供缓冲区功能

1、字符流
实例一:字符流写入

package ioTest;import java.io.FileWriter;
import java.io.IOException;public class Main {public static void main (String args[]){//创建要操作的文件路径和名称 //其中,系统相关的分隔符,Linux下为:/  Windows下为:\\ String path = "E:\\java\\demo.txt";//由于IO操作会抛出异常,因此在try语句块的外部定义FileWriter的引用 FileWriter w = null; try { //以path为路径创建一个新的FileWriter对象 //如果需要追加数据,而不是覆盖,则使用FileWriter(path,true)构造方法 w = new FileWriter(path); //将字符串写入到流中,\r\n表示换行想有好的 w.write("我喜欢你!\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:字符流的读取

package ioTest;import java.io.FileReader;
import java.io.IOException;public class Main {public static void main (String args[]){//创建要操作的文件路径和名称 //其中,分隔符,Linux下为:/  Windows下为:\\ String path = "E:\\java\\love.txt";FileReader r = null; try { r = new FileReader(path); //方式一:读取单个字符的方式 //每读取一次,向下移动一个字符单位 
//            int temp1 = r.read(); 
//            System.out.print((char)temp1); 
//            int temp2 = r.read(); 
//            System.out.print((char)temp2); 
//            int temp3 = r.read(); 
//            System.out.print((char)temp3); 
//            int temp4 = r.read(); 
//            System.out.print((char)temp4);
//            int temp5 = r.read(); 
//            System.out.print((char)temp5);
//                          //方式二:循环读取 //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:文本文件的复制

package ioTest;import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;public class Main {public static void main (String args[]){//创建要操作的文件路径和名称 //其中,分隔符,Linux下为:/  Windows下为:\\ String path = "E:\\java\\love.txt";String path1 = "E:\\java\\you.txt";FileReader r = null; FileWriter w = null; try { r = new FileReader(path); w = new FileWriter(path1); //方式一:单个字符写入 /* 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:利用字符流的缓冲区来进行文本文件的复制

package ioTest;import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;public class Main {public static void main (String args[]){//创建要操作的文件路径和名称 //其中,分隔符,Linux下为:/  Windows下为:\\ String path = "E:\\java\\love.txt";String path1 = "E:\\java\\you.txt";FileReader r = null; FileWriter w = null;//创建缓冲区的引用 BufferedReader br = null; BufferedWriter bw = null; try { r = new FileReader(path); w = new FileWriter(path1); //创建缓冲区对象 //将需要提高效率的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下则是\n bw.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(); } } } } 
}

效果:
这里写图片描述

2、字节流
实例4:字节流的写入

package ioTest;import java.io.FileOutputStream;
import java.io.IOException;public class Main {public static void main (String args[]){//创建要操作的文件路径和名称 //其中,分隔符,Linux下为:/  Windows下为:\\ String path = "E:\\java\\love.txt";FileOutputStream o = null; try { o = new FileOutputStream(path); String str = "蓝瘦,香菇!\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:字节流的读取

package ioTest;import java.io.FileInputStream;
import java.io.IOException;public class Main {public static void main (String args[]){//创建要操作的文件路径和名称 //其中,分隔符,Linux下为:/  Windows下为:\\ String path = "E:\\java\\love.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:二进制文件的复制

package ioTest;import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;public class Main {public static void main (String args[]){//创建要操作的文件路径和名称 //其中,分隔符,Linux下为:/  Windows下为:\\ String path = "E:\\java\\演员.mp3";String path1 = "E:\\java\\演员1.mp3";FileInputStream i = null; FileOutputStream o = null; try { i = new FileInputStream(path); o = new FileOutputStream(path1); //循环的方式读入写出文件,从而完成复制 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:利用字节流的缓冲区进行二进制文件的复制

package ioTest;import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;public class Main {public static void main (String args[]){//创建要操作的文件路径和名称 //其中,分隔符,Linux下为:/  Windows下为:\\ String path = "E:\\java\\演员.mp3";String path1 = "E:\\java\\演员2.mp3";FileInputStream i = null; FileOutputStream o = null; BufferedInputStream bi = null; BufferedOutputStream bo = null; try { i = new FileInputStream(path); o = new FileOutputStream(path1); 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,视频文件等

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

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

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

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

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

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

这篇关于Java—IO流详解(一)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/940570

相关文章

java io包

InputStream InputStream 是 Java I/O 中所有输入流的抽象基类,它定义了读取字节流的基本方法。InputStream 类提供了许多子类,用于从不同的数据源读取数据,如文件、网络连接、内存等。 InputStream 提供了以下常用的方法: int read(): 从输入流中读取下一个字节的数据。如果已经到达流的末尾,则返回 -1。int read(byte[

文件IO和 /

前言 文件流对象和>>及<<的匹配老搞混,恼火。现在做个了断。以下面这个例子具体分析。 情况描述 ifstream继承了istream,因此可以将ifstream对象作为参数传递给形参为istream的函数。编写一个示例程序验证。 我的错误代码: /* 程序往文件1中写入字符串"GoodBye" */#include <iostream>#include <f

基本文件IO的实现

前言 有时要写个简单的文件IO程序,却忘了步骤。郁闷。这次整一个标准的模板代码,忘了来查就好了。 示例1:(如下代码从标准输入获取字符串并写入到文件) #include <iostream>#include <fstream>using namespace std;int main(){fstream out;string file1; cout << "Which file

Java找不到包解决方案

在跟着教程写Spingboot后端项目时,为了加快效率,有时候有的实体文件可以直接粘贴到目录中,此时运行项目会出现Java找不到包的情况,即无法找到导入的实体文件,这是项目没有更新的原因。解决方法:         刷新Maven:         点击如下循环,即刷新Maven的包。之后便可以更新之前导入的包。

javascript一些常用转换方法

1、容量单位转换 function bytesToSize(bytes) {if (bytes === 0) return '0 B';var k = 1000, // or 1024sizes = ['B', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'],i = Math.floor(Math.log(bytes) / Math.log(k))

容器化管理SpringBoot项目:在用jar包制作镜像的时候遇到的错误记录

在容器化管理SpringBoot项目,进行到“用jar包制作镜像”一步时,遇到的error真的是一环接着一环,这里就记录一下,一套流程下来遇到的error,以及一些我的解决方法: ERROR: "docker buildx build" requires exactly 1 argument. See 'docker buildx build --help'.  Usage:  docker

【转】JavaScript——连续赋值与求值顺序(var a.x=a={n:2})

例题 var a = {n: 1};var b = a;a.x = a = { n: 2 };alert(a.x); // undefinedalert(b.x); // [object Object] 解析: a.x = a = { n: 2 }; . 操作符的优先级比 = 高,= 从右向左解析,a.x -> 这时候的a还是原来的a, a = {n: 1,x: undefin

通过 Java 操作 redis -- list 列表基本命令

目录 使用命令 lpush,lrange,rpush 使用命令 lpop 和 rpop 使用命令 blpop,brpop 使用命令 llen 关于 redis list 列表类型的相关命令推荐看Redis - list 列表 要想通过 Java 操作 redis,首先要连接上 redis 服务器,推荐看通过 Java 操作 redis -- 连接 redis

Javaweb第五次作业

poet数据库sql语言 create table poet(id int unsigned primary key auto_increment comment 'ID',name varchar(10) not null comment '姓名',gender tinyint unsigned not null comment '性别, 说明: 1 男, 2 女',dynasty varch

Spring自动装配:解析原理与实践

在Spring框架中,自动装配是一种强大的特性,它能够根据一定的规则自动地将bean装配到Spring容器中,从而简化了配置和开发过程。本文将深入探讨Spring自动装配的原理和实践,帮助程序员更好地理解和应用这一重要特性。 1. 什么是自动装配? 在传统的Spring开发中,我们需要在XML配置文件或Java配置类中显式地定义bean的依赖关系和装配规则。而自动装配则是一种更加方便的方式,它