Java基础学习笔记(十六)—— IO流

news2024/11/28 23:46:01

IO流

  • 1 IO流
    • 1.1 IO流概述
    • 1.2 IO流的分类
    • 1.3 IO流的使用场景
  • 2 File类
    • 2.1 File类概述
    • 2.2 File类构造方法
    • 2.3 File类常用方法
    • 2.4 File类案例
  • 3 字节流
    • 3.1 字节流写数据
    • 3.2 字节流写数据的三种方式
    • 3.3 字节流写数据加异常处理
    • 3.4 字节流读数据
    • 3.5 字节流复制文件
  • 4 字节缓冲流
    • 4.1 字节缓冲流概述
    • 4.2 字节缓冲流构造方法
    • 4.3 小结
  • 5 字符流
    • 5.1 字符流概述
    • 5.2 字符串的编码解码
    • 5.3 字符流写数据
    • 1.4 字符流读数据
    • 5.5 字符流案例
  • 6 字符缓冲流
    • 6.1 构造方法
    • 6.2 特有方法
    • 6.3 缓冲流案例

1 IO流

之前存储的数据是不能够永久化存储的,只要代码运行结束,所有数据都会丢失。而想要数据永久话存储就必须要用到IO,可以跟本地硬盘交互。

1.1 IO流概述

  • IO:输入/输出(Input/Output)
    • I表示input,是数据从硬盘进内存的过程,称为读
    • O表示output,是数据从内存到硬盘的过程,称为写
  • 流:是一种抽象概念,是对数据传输的总称,也就是说数据在设备间的传输称为流,流的本质是数据传输

IO的数据传输,可以看做是一种数据的流动,按照流动的方向,以内存为参照物,进行读写操作。

1.2 IO流的分类

  • 按照数据的流向
    • 输入流:读数据
    • 输出流:写数据
  • 按照数据类型来分
    • 字节流
      • 字节输入流
      • 字节输出流
    • 字符流
      • 字符输入流
      • 字符输出流

在这里插入图片描述

1.3 IO流的使用场景

  • 如果操作的是纯文本文件,优先使用字符流
  • 如果操作的是图片、视频、音频等二进制文件,优先使用字节流
  • 如果不确定文件类型,优先使用字节流,字节流是万能的流

纯文本文件:用Windows自带的记事本打开能读懂的文件就是纯文本文件,txt文件就直接是纯文本文件

2 File类

2.1 File类概述

File类

  • 它是文件和目录路径名的抽象表示
  • 文件和目录是可以通过File封装成对象的
  • 对于File而言,其封装的并不是一个真正存在的文件,仅仅是一个路径名而已。它可以是存在的,也可以是不存在的,将来是要通过具体的操作把这个路径的内容转换为具体存在的
  • 在读写数据时告诉虚拟机要操作的(文件/文件夹)在哪
  • 对(文件/文件夹)进行操作,包括创建、删除等

2.2 File类构造方法

在这里插入图片描述

public static void main(String[] args) {
	//method1();
	//method2();
	//method3();
}

private static void method3() {
	//File​(File parent, String child)      从父抽象路径名和子路径名字符串创建新的File实例
	File file1 = new File("C:\\it");
	String path = "a.txt";
	File file = new File(file1,path);
	System.out.println(file);//C:\it\a.txt
}

private static void method2() {
	//File​(String parent, String child)    从父路径名字符串和子路径名字符串创建新的File实例
	String path1 = "C:\\it";
	String path2 = "a.txt";
	File file = new File(path1,path2);//把两个路径拼接.
	System.out.println(file);//C:\it\a.txt
}

private static void method1() {
	//File​(String pathname)        通过将给定的路径名字符串转换为抽象路径名来创建新的File实例
	String path = "C:\\it\\a.txt";
	File file = new File(path);
	//问题:为什么要把字符串表示形式的路径变成File对象?
	//就是为了使用File类里面的方法.
}

注意:

  • 绝对路径
    • 是一个完整的路径,从盘符开始
  • 相对路径
    • 是一个简化的路径,相对当前项目下的路径
public static void main(String[] args) {
    // 是一个完整的路径,从盘符开始
    File file1 = new File("D:\\it\\a.txt");

    // 是一个简化的路径,从当前项目根目录开始
    File file2 = new File("a.txt");
    File file3 = new File("模块名\\a.txt");
}

2.3 File类常用方法

  1. 创建功能

在这里插入图片描述

public static void main(String[] args) throws IOException {
	//public boolean createNewFile()    创建一个新的空的文件
			//注意点:
				//1.如果文件存在,那么创建失败,返回false
				//2.如果文件不存在,那么创建成功,返回true
				//3.createNewFile方法不管调用者有没有后缀名,只能创建文件.

	//public boolean mkdir()            创建一个单级文件夹
			//注意点:
				//1.只能创建单级文件夹,不能创建多级文件夹
				//2.不管调用者有没有后缀名,只能创建单级文件夹

	//public boolean mkdirs()           创建一个多级文件夹
			//注意点:
				//1,可以创建单级文件夹,也可以创建多级文件夹
				//2.不管调用者有没有后缀名,只能创建文件夹

	//疑问:
		//既然mkdirs能创建单级,也能创建多级.那么mkdir还有什么用啊? 是的


	//method1();
	//method2();

	File file = new File("C:\\it\\aaa.txt");
	boolean result = file.mkdirs();
	System.out.println(result);
}

private static void method2() {
	File file = new File("C:\\it\\aaa.txt");
	boolean result = file.mkdir();
	System.out.println(result);
}

private static void method1() throws IOException {
	File file1 = new File("C:\\it\\aaa");
	boolean result1 = file1.createNewFile();
	System.out.println(result1);
}
  1. 删除功能

在这里插入图片描述

//注意点:
	//1.不走回收站的.
	//2.如果删除的是文件,那么直接删除.如果删除的是文件夹,那么能删除空文件夹
	//3.如果要删除一个有内容的文件夹,只能先进入到这个文件夹,把里面的内容全部删除完毕,才能再次删除这个文件夹
//简单来说:
	//只能删除文件和空文件夹.
public static void main(String[] args) {
	//method1();
	File file = new File("C:\\it");
	boolean result = file.delete();
	System.out.println(result);
}

private static void method1() {
	File file = new File("C:\\it\\a.txt");
	boolean result = file.delete();
	System.out.println(result);
}
  1. 判断功能

在这里插入图片描述

public static void main(String[] args) {
	//method1();
	//method2();
	
private static void method2() {
	File file = new File("a.txt");
	boolean result = file.exists();
	System.out.println(result);
}

private static void method1() {
	File file = new File("C:\\it\\a.txt");
	boolean result1 = file.isFile();
	boolean result2 = file.isDirectory();
	System.out.println(result1);
	System.out.println(result2);
}
  1. 获取功能

在这里插入图片描述

public static void main(String[] args) {
   File file = new File("D:\\aaa");
   System.out.println(f.getAbsolutePath());
   System.out.println(f.getPath());
   System.out.println(f.getName());

	File[] files = file.listFiles();//返回值是一个File类型的数组
	System.out.println(files.length);
	for (File path : files) {
		System.out.println(path);
	}
}

注意:

  • 当调用者不存在时,listFiles方法返回null
  • 当调用者是一个文件时,listFiles方法返回null
  • 当调用者是一个空文件夹时,listFiles方法返回一个长度为0的数组
  • 当调用者是一个有内容的文件夹时,进入文件夹,listFiles方法会获取这个文件夹里面所有的文件和文件夹的File对象,并把这些File对象都放在一个数组中返回。包括隐藏文件和隐藏文件夹都可以获取。
  • 当调用者是一个有权限才能进入的文件夹时,listFiles方法返回null

2.4 File类案例

  1. 在当前模块下的aaa文件夹中创建一个a.txt文件
public static void main(String[] args) throws IOException {
   /* File file = new File("filemodule\\aaa\\a.txt");
	file.createNewFile();*/
	//注意点:文件所在的文件夹必须要存在.

	File file = new File("filemodule\\aaa");
	if(!file.exists()){
		//如果文件夹不存在,就创建出来
		file.mkdirs();
	}
	File newFile = new File(file,"a.txt");
	newFile.createNewFile();
}
  1. 删除一个多级文件夹
public static void main(String[] args) {
	//delete方法
	//只能删除文件和空文件夹.
	//如果现在要删除一个有内容的文件夹?
	//先删掉这个文件夹里面所有的内容.
	//最后再删除这个文件夹
	File src = new File("C:\\Users\\apple\\Desktop\\src");
	deleteDir(src);
}

private static void deleteDir(File src) {
	//先删掉这个文件夹里面所有的内容.
	//递归 方法在方法体中自己调用自己.
	//注意: 可以解决所有文件夹和递归相结合的题目
	//1.进入 --- 得到src文件夹里面所有内容的File对象.
	File[] files = src.listFiles();
	//2.遍历 --- 因为我想得到src文件夹里面每一个文件和文件夹的File对象.
	for (File file : files) {
		if(file.isFile()){
			//3.判断 --- 如果遍历到的File对象是一个文件,那么直接删除
			file.delete();
		}else{
			//4.判断
			//递归
			deleteDir(file);//参数一定要是src文件夹里面的文件夹File对象
		}
	}
	//最后再删除这个文件夹
	src.delete();
}
  1. 统计一个文件夹中,每种文件出现的次数
public static void main(String[] args) {
	//统计 --- 定义一个变量用来统计. ---- 弊端:同时只能统计一种文件
	//利用map集合进行数据统计,键 --- 文件后缀名  值 ----  次数

	File file = new File("filemodule");
	HashMap<String, Integer> hm = new HashMap<>();
	getCount(hm, file);
	System.out.println(hm);
}

private static void getCount(HashMap<String, Integer> hm, File file) {
	File[] files = file.listFiles();
	for (File f : files) {
		if(f.isFile()){
			String fileName = f.getName();
			String[] fileNameArr = fileName.split("\\.");
			if(fileNameArr.length == 2){
				String fileEndName = fileNameArr[1];
				if(hm.containsKey(fileEndName)){
					//已经存在
					//将已经出现的次数获取出来
					Integer count = hm.get(fileEndName);
					//这种文件又出现了一次.
					count++;
					//把已经出现的次数给覆盖掉.
					hm.put(fileEndName,count);
				}else{
					//不存在
					//表示当前文件是第一次出现
					hm.put(fileEndName,1);
				}
			}
		}else{
			getCount(hm,f);
		}
	}
}

3 字节流

3.1 字节流写数据

  • 字节流抽象基类
    • InputStream:这个抽象类是表示字节输入流的所有类的超类
    • OutputStream:这个抽象类是表示字节输出流的所有类的超类
    • 子类名特点:子类名称都是以其父类名作为子类名的后缀
  • 字节输出流
    • FileOutputStream(String name):创建文件输出流以指定的名称写入文件
  • 使用字节输出流写数据的步骤
    • 创建字节输出流对象(调用系统功能创建了文件,创建字节输出流对象,让字节输出流对象指向文件)
    • 调用字节输出流对象的写数据方法
    • 释放资源(关闭此文件输出流并释放与此流相关联的任何系统资源)
public static void main(String[] args) throws IOException {
	//1.创建字节输出流的对象 --- 告诉虚拟机我要往哪个文件中写数据了
	FileOutputStream fos = new FileOutputStream("D:\\a.txt");
	//FileOutputStream fos = new FileOutputStream(new File("D:\\a.txt"));

	//2,写数据,传递一个整数时,那么实际上写到文件中的,是这个整数在码表中对应的那个字符.
	fos.write(97); //会在文件中写入a

	//3,释放资源,告诉操作系统,我现在已经不要再用这个文件了
	fos.close();
}

注意:

  • 如果文件不存在,会帮我们自动创建出来.
  • 如果文件存在,会把文件清空.

3.2 字节流写数据的三种方式

在这里插入图片描述

public static void main(String[] args) throws IOException {
    FileOutputStream fos = new FileOutputStream("bytestream\\a.txt");
  /*fos.write(97);
    fos.write(98);
    fos.write(99);*/
    
    /* byte [] bys = {97,98,99};
    fos.write(bys);*/

    byte [] bys = {97,98,99,100,101,102,103};
    fos.write(bys,1,2); //98,99即写入bc
    
    fos.close();
}

那么,现在有两个小问题

  • 字节流写数据如何实现换行?
    • windows:\r\n
    • linux:\n
    • mac:\r
    • getBytes()是字符串的一个方法,可以将字符串转换为字节
  • 字节流写数据如何实现追加写入?
    • public FileOutputStream(String name,boolean append)
    • 创建文件输出流以指定的名称写入文件。如果第二个参数为true ,则字节将写入文件的末尾而不是开头
public static void main(String[] args) throws IOException {
	//第二个参数就是续写开关,如果没有传递,默认就是false,
	//表示不打开续写功能,那么创建对象的这行代码会清空文件.

	//如果第二个参数为true,表示打开续写功能
	//那么创建对象的这行代码不会清空文件.
	FileOutputStream fos = new FileOutputStream("bytestream\\a.txt",true);

	fos.write(97);
	//加一个换行
	fos.write("\r\n".getBytes());
	fos.write(98);
	//加一个换行
	fos.write("\r\n".getBytes());
	fos.write(99);
	//加一个换行
	fos.write("\r\n".getBytes());
	fos.write(100);
	//加一个换行
	fos.write("\r\n".getBytes());
	fos.write(101);
	//加一个换行
	fos.write("\r\n".getBytes());

	fos.close();
}

3.3 字节流写数据加异常处理

有如下代码:

try {
    FileOutputStream fos = new FileOutputStream("a.txt");
    fos.write(97);
    fos.close();
}catch (IOException e){
    e.printStackTrace();
}

我们如何操作才能让close方法一定执行呢?

异常处理的标准格式:

try{
	可能出现异常的代码;
}catch(异常类名 变量名){
	异常的处理代码;
}finally{
	执行所有清除操作; // 在异常处理时提供finally块来执行所有的清除操作,比如IO流的释放资源,被finally控制的语句一定会执行,除非JVM退出
}

加异常处理后的代码如下:

public static void main(String[] args) {
	FileOutputStream fos = null;
	try {
		//System.out.println(2/0);
		fos = new FileOutputStream("D:\\a.txt");
		fos.write(97);
	}catch(IOException e){
	   e.printStackTrace();
	}finally {
		//finally语句里面的代码,一定会被执行.
		if(fos != null){
			try {
				fos.close();
			} catch (IOException e) {
				e.printStackTrace();
			}
		}
	}
}

3.4 字节流读数据

字节输入流

  • FileInputStream(String name):通过打开与实际文件的连接来创建一个FileInputStream,该文件由文件系统中的路径名name命名

字节输入流读取数据的步骤

  • 创建字节输入流对象
  • 调用字节输入流对象的读数据方法
  • 释放资源
  1. 一次读一个字节数据
public static void main(String[] args) throws IOException {
	//如果文件存在,那么就不会报错.
	//如果文件不存在,那么就直接报错.
	FileInputStream fis = new FileInputStream("bytestream\\a.txt");

	int read = fis.read();
	//一次读取一个字节,返回值就是本次读到的那个字节数据.
	//也就是字符在码表中对应的那个数字.
	//如果我们想要看到的是字符数据,那么一定要强转成char

	System.out.println(read); // 97
	System.out.println((char)read); //a

	//释放资源
	fis.close();
}
  1. 一次读多个字节数据
public static void main(String[] args) throws IOException {
	FileInputStream fis = new FileInputStream("bytestream\\a.txt");
	//文件中多个字节我怎么办?
	/*while(true){
		int i1 = fis.read(); 内容读取结束读取到空格时返回-1
		System.out.println(i1); 
	}*/ 

	int b;
	while ((b = fis.read())!=-1){
		System.out.println((char) b);
	}
	fis.close();
}

3.5 字节流复制文件

  1. 小文件复制

将C:\it\a.jpg的文件复制到模块bytestream下

public static void main(String[] args) throws IOException {
	//创建了字节输入流,准备读数据.
	FileInputStream fis = new FileInputStream("C:\\it\\a.jpg");
	//创建了字节输出流,准备写数据.
	FileOutputStream fos = new FileOutputStream("bytestream\\a.jpg");

	int b;
	while((b = fis.read())!=-1){
		fos.write(b);
	}

	fis.close();
	fos.close();
}
  1. 大文件复制

对于大文件的复制问题,字节流通过创建字节数组,可以一次读写多个数据

一次读一个字节数组的方法:

  • public int read(byte[] b):从输入流读取最多b.length个字节的数据
  • 返回的是读入缓冲区的总字节数,也就是实际的读取字节个数
public static void main(String[] args) throws IOException {
	FileInputStream fis = new FileInputStream("C:\\itheima\\a.avi");
	FileOutputStream fos = new FileOutputStream("bytestream\\a.avi");

	byte [] bytes = new byte[1024];// 该字节数组的大小是1024字节
	int len;//本次读到的有效字节个数 --- 这次读了几个字节

	while((len = fis.read(bytes))!=-1){  // 循环读取
		fos.write(bytes,0,len);//0索引开始,读取len个字节
	}

	fis.close();
	fos.close();
}

4 字节缓冲流

4.1 字节缓冲流概述

  • BufferedOutputStream:该类实现缓冲输出流,通过设置这样的输出流,应用程序可以向底层输出流写入字节,而不必为写入的每个字节导致底层系统的调用
  • BufferedInputStream:创建BufferedInputStream将创建一个内部缓冲区数组,当从流中读取或跳过字节时,内部缓冲区将根据需要从所包含的输入流中重新填充,一次很多字节

4.2 字节缓冲流构造方法

在这里插入图片描述

  1. 一次读写一个字节
public static void main(String[] args) throws IOException {
	//就要利用缓冲流去拷贝文件

	//创建一个字节缓冲输入流
	//在底层创建了一个默认长度为8192的字节数组。
	BufferedInputStream bis = new BufferedInputStream(new FileInputStream("bytestream\\a.avi"));
	//创建一个字节缓冲输出流
	//在底层也创建了一个默认长度为8192的字节数组。
	BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("bytestream\\copy.avi"));

	int b;
	while((b = bis.read()) != -1){
		bos.write(b);
	}

	//方法的底层会把字节流给关闭。
	bis.close();
	bos.close();
}
  1. 一次读写一个字节数组
public static void main(String[] args) throws IOException {
	//缓冲流结合数组,进行文件拷贝

	//创建一个字节缓冲输入流
	BufferedInputStream bis = new BufferedInputStream(new FileInputStream("bytestream\\a.avi"));

	//创建一个字节缓冲输出流
	BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("bytestream\\copy.avi"));

	byte [] bytes = new byte[1024];
	int len;
	while((len = bis.read(bytes)) != -1){
		bos.write(bytes,0,len);
	}

	bis.close();
	bos.close();
}

为什么构造方法需要的是字节流,而不是具体的文件或者路径呢?

  • 字节缓冲流仅仅提供缓冲区,而真正读写数据还是得依靠基本的字节流对象进行操作

4.3 小结

字节流:

  • 可以操作(拷贝)所有类型的文件

字节缓冲流:

  • 可以提高效率;
  • 不能直接操作文件,需要传递字节流

拷贝文件的四种方式:

  • 字节流一次读写一个字节
  • 字节流一次读写一个字节数组
  • 字节缓冲流一次读写一个字节
  • 字节缓冲流一次读写一个字节数组

5 字符流

5.1 字符流概述

既然字节流可以操作所有文件,为什么要学习字符流?

  • 如果利用字节流,把文本文件中的中文,读取到内存中,有可能出现乱码
  • 如果利用字节流,把中文写到文本文件中,也有可能出现乱码

为什么字节流读取文本文件,可能会出现乱码?

  • 因为字节流一次读一个字节,而不管GBK还是UTF-8一个中文都是多个字节,用字节流每次只能读其中的一部分,所以会出现乱码的问题

由于字节流操作中文不是特别的方便,所以Java就提供字符流(字符流 = 字节流 + 编码表

  • 计算某种规则,将字符变成二进制,再存储到计算机中,称为编码
  • 按照同样的规则,将存储在计算机中的二进制数解析显示出来,称为解码
  • 编码和解码的方式必须一致,否则会导致乱码

Windows默认使用码表为:GBK,一个中文两个字节
IDEA和以后工作默认使用Unicode的UTF-8编码格式,一个中文三个字节
不管是在哪张码表中,中文的第一个字节一定是负数

注意:

  • 想要进行拷贝,一律使用字节流或者字节缓冲流
  • 想要把文本文件中的数据读到内存中,使用字符输入流
  • 想要把内存中的数据写到文本文件中,使用字符输出流

5.2 字符串的编码解码

在这里插入图片描述

public static void main(String[] args) throws UnsupportedEncodingException {
    method1();
    method2();
}

private static void method2() throws UnsupportedEncodingException {
    byte[] bytes =  {-28, -67, -96, -27, -91, -67, -28, -72, -106, -25, -107, -116};

    //利用默认的UTF-8进行解码
    String s1 = new String(bytes);
    System.out.println(s1);//你好世界

    //利用指定的编码进行解码
    String s2 = new String(bytes,"UTF-8");
    System.out.println(s2);//你好世界
}

private static void method1() throws UnsupportedEncodingException {
    String s  = "你好世界";
    //利用idea默认的UTF-8将中文编码为一系列的字节
    byte[] bytes1 =  s.getBytes();
    System.out.println(Arrays.toString(bytes1)); // [-28, -67, -96, -27, -91, -67, -28, -72, -106, -25, -107, -116]

    byte[] bytes2 =  s.getBytes("UTF-8");
    System.out.println(Arrays.toString(bytes1)); // [-28, -67, -96, -27, -91, -67, -28, -72, -106, -25, -107, -116]
}

5.3 字符流写数据

Writer: 用于写入字符流的抽象父类
FileWriter: 用于写入字符流的常用子类

构造方法:

在这里插入图片描述

成员方法:

在这里插入图片描述

public static void main(String[] args) throws IOException {
    //创建字符输出流的对象
    //FileWriter fw = new FileWriter(new File("charstream\\a.txt"));
    FileWriter fw = new FileWriter("charstream\\a.txt");


    //写一个字符
    fw.write(97);
    fw.write(98);
    fw.write(99);


    //写出一个字符数组
    char [] chars1 = {97,98,99,100,101};
    fw.write(chars1);


    //写出字符数组的一部分
    char [] chars2 = {97,98,99,100,101};
    fw.write(chars2,0,3);


    //写一个字符串
    String line1 = "黑马程序员abc";
    fw.write(line1);


    //写一个字符串的一部分
    String line2 = "黑马程序员abc";
    fw.write(line2, 0, 2);

    //释放资源
    fw.close();
}

注意:

  • 在创建字符输出流对象时,如果文件存在就清空,如果文件不存在就创建,但是要保证父级路径存在
  • 写数据时,写出int类型的整数,实际写出的是整数在码表上对应的字母,写出字符串数据,是把字符串本身原样输出

刷新和关闭的方法:

在这里插入图片描述

1.4 字符流读数据

Reader: 用于读取字符流的抽象父类
FileReader: 用于读取字符流的常用子类

  • 构造方法

在这里插入图片描述

  • 成员方法

在这里插入图片描述

public static void main(String[] args) throws IOException {
    //创建字符输入流的对象
    // FileReader fr = new FileReader(new File("charstream\\a.txt"));
    FileReader fr = new FileReader("charstream\\a.txt");

    //读取数据
    
    //一次读取一个字符
/*        int ch;
    while((ch = fr.read()) != -1){
        System.out.println((char) ch);
    }*/


    //一次读取多个字符。
    //创建一个数组
    char [] chars = new char[1024];
    int len;
    //read方法还是读取,但是是一次读取多个字符
    //他把读到的字符都存入到chars数组。
    //返回值:表示本次读到了多少个字符。
    while((len = fr.read(chars))!=-1){
        System.out.println(new String(chars,0,len));
    }

    //释放资源
    fr.close();
}

5.5 字符流案例

  • 案例需求:
    • 将键盘录入的用户名和密码保存到本地实现永久化存储
  • 实现步骤:
    • 获取用户输入的用户名和密码
    • 将用户输入的用户名和密码写入到本地文件中
    • 关流,释放资源
  • 示例代码:
public static void main(String[] args) throws IOException {
	//将键盘录入的用户名和密码保存到本地实现永久化存储
	//要求:用户名独占一行,密码独占一行

	//分析:
	//1,实现键盘录入,把用户名和密码录入进来
	Scanner sc = new Scanner(System.in);
	System.out.println("请录入用户名");
	String username = sc.next();
	System.out.println("请录入密码");
	String password = sc.next();

	//2.分别把用户名和密码写到本地文件。
	FileWriter fw = new FileWriter("charstream\\a.txt");
	//将用户名和密码写到文件中
	fw.write(username);
	//表示写出一个回车换行符 windows \r\n  MacOS \r  Linux \n
	fw.write("\r\n");
	fw.write(password);
	//刷新流
	fw.flush();
	//释放资源
	fw.close();
}

6 字符缓冲流

6.1 构造方法

在这里插入图片描述

  • BufferedWriter:将文本写入字符输出流,缓冲字符,以提供单个字符,数组和字符串的高效写入,可以指定缓冲区大小,或者可以接受默认大小。默认值足够大,可用于大多数用途

  • BufferedReader:从字符输入流读取文本,缓冲字符,以提供字符,数组和行的高效读取,可以指定缓冲区大小,或者可以使用默认大小。 默认值足够大,可用于大多数用途

public static void main(String[] args) throws IOException {
    read();
    write();
}

private static void write() throws IOException {
    //字符缓冲输出流
    BufferedWriter bw = new BufferedWriter(new FileWriter("charstream\\a.txt"));

    //写出数据
    //实际写出的是97对应的字符a
    bw.write(97);
    bw.write("\r\n");

    //实际写出的是97 - 101 对应的字符 abcde
    char [] chars = {97,98,99,100,101};
    bw.write(chars);
    bw.write("\r\n");

    //实际写的是abc
    bw.write(chars,0,3);
    bw.write("\r\n");

    //会把字符串的内容原样写出
    bw.write("我是一个程序员");
    bw.write("\r\n");

    //会把字符串的一部分写出 abcde
    String line = "abcdefg";
    bw.write(line,0,5);
    
    bw.flush();
    bw.close();

}

private static void read() throws IOException {
    //字符缓冲输入流
    BufferedReader br = new BufferedReader(new FileReader("charstream\\a.txt"));

    //读取数据
    char [] chars = new char[1024];
    int len;
    while((len = br.read(chars)) != -1){
        System.out.println(new String(chars,0,len));
    }

    br.close();
}

6.2 特有方法

BufferedWriter:

在这里插入图片描述

public static void main(String[] args) throws IOException {
	//字符缓冲流的特有功能
	//字符缓冲输出流BufferedWrite : newLine  跨平台的换行符


	//创建对象
	BufferedWriter bw = new BufferedWriter(new FileWriter("charstream\\a.txt"));


	//写出数据
	bw.write("程序员666");
	//跨平台的回车换行
	bw.newLine();
	bw.write("abcdef");
	//跨平台的回车换行
	bw.newLine();
	bw.write("-------------");

	//刷新流
	bw.flush();
	//释放资源
	bw.close();
}

BufferedReader:

在这里插入图片描述

public static void main(String[] args) throws IOException {
    //字符缓冲流的特有功能
    //字符缓冲输入流BufferedReader: readLine 读一整行

    //创建对象
    BufferedReader br = new BufferedReader(new FileReader("charstream\\a.txt"));


/*        //读取数据
    String line1 = br.readLine();
    String line2 = br.readLine();
    String line3 = br.readLine();
    //在之前,如果读不到数据,返回-1
    //但是readLine如果读不到数据返回null
    String line4 = br.readLine();
    System.out.println(line1);
    System.out.println(line2);
    System.out.println(line3);
    System.out.println(line4);*/

    //使用循环来进行改进
    String line;
    //可以读取一整行数据。一直读,读到回车换行为止。
    //但是他不会读取回车换行符。
    while((line = br.readLine()) != null){
        System.out.println(line);
    }
    
    // 释放资源
    br.close();
}

6.3 缓冲流案例

字符缓冲流操作文件中数据排序案例

  • 案例需求
    • 使用字符缓冲流读取文件中的数据,排序后再次写到本地文件
  • 实现步骤
    • 将文件中的数据读取到程序中
    • 对读取到的数据进行处理
    • 将处理后的数据添加到集合中
    • 对集合中的数据进行排序
    • 将排序后的集合中的数据写入到文件中
  • 代码实现
public static void main(String[] args) throws IOException {
	//需求:读取文件中的数据,排序后再次写到本地文件
	//分析:
	//1.要把文件中的数据读取进来。
	BufferedReader br = new BufferedReader(new FileReader("charstream\\sort.txt"));
	//输出流一定不能写在这里,因为会清空文件中的内容
	//BufferedWriter bw = new BufferedWriter(new FileWriter("charstream\\sort.txt"));

	String line = br.readLine();
	System.out.println("读取到的数据为" + line);
	br.close();

	//2.按照空格进行切换
	String[] split = line.split(" ");//9 1 2 5 3 10 4 6 7 8
	//3.把字符串类型的数组变成int类型
	int [] arr = new int[split.length];
	//遍历split数组,可以进行类型转换。
	for (int i = 0; i < split.length; i++) {
		String smallStr = split[i];
		//类型转换
		int number = Integer.parseInt(smallStr);
		//把转换后的结果存入到arr中
		arr[i] = number;
	}
	//4.排序
	Arrays.sort(arr);
	System.out.println(Arrays.toString(arr));


	//5.把排序之后结果写回到本地 1 2 3 4...
	BufferedWriter bw = new BufferedWriter(new FileWriter("charstream\\sort.txt"));
	//写出
	for (int i = 0; i < arr.length; i++) {
		bw.write(arr[i] + " ");
		bw.flush();
	}
	//释放资源
	bw.close();
}

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

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

相关文章

String,StringBuilder,StringBuffer源码分析

声明&#xff1a;本文使用JDK1.8 在Java中&#xff0c;对于字符串的操作有这三种&#xff1a;String、StringBuilder、StringBuffer。这三者的效率是&#xff1a;StringBuilder > StringBuffer > String。 String a "abc"; a a "d"; System.out.…

好看的滤镜怎么调?来看看今日的分享

通常我们会将拍完的照片进行美化操作&#xff0c;那么美化照片的一个重要步骤就是添加滤镜了&#xff0c;但该如何给照片加滤镜&#xff0c;却是很多人不知道的&#xff0c;滤镜可不是随随便便添加就好的&#xff0c;它不仅仅需要匹配我们的照片&#xff0c;还得对添加的滤镜参…

Maix Bit(K210)保姆级入门上手教程---外设基本使用

Maix Bit&#xff08;K210&#xff09;保姆级入门上手教程系列 Maix Bit&#xff08;K210&#xff09;保姆级入门上手教程—环境搭建 这是K210快速上手系列文章&#xff0c;主要内容是&#xff0c;介绍K210的基本外设的使用、通过简单介绍一两个基本的硬件使用来掌握K210的外设…

Elasticsearch连续剧之分词器

目录一、前言二、默认分词器三、IK分词器1.主要算法2.安装IK分词器2.1 关闭es服务2.2 上传ik分词器到虚拟机2.3 解压2.4 启动ES服务2.5 测试分词器效果2.6 IK分词器词典四、拼音分词器1.安装2.测试分词效果五、自定义分词器1.创建自定义分词器2.测试一、前言 ES文档的数据拆分…

如何使用OpenDRIVE

文章目录OpenDRIVE Notes#1 前言#2 OpenDRIVE结构#2.1 Road#2.1.1 道路属性#2.1.2 道路联接#2.1.3 参考线#2.2 laneSection#2.3 laneOffset#2.4 junction#2.4.1 路口的联接#2.5 poly3(三次多项式)#3 解析#3.1 数据结构#3.1.1 ID#3.1.2 Point#4 构建topo#5 邻接点#6 路径规划#7 …

计算机图形学实习教程之基本图形的生成(二维图形裁剪Cohen-Sutherland算法+图形平移算法+图形旋转算法),利用C#实现,附源码

环境&#xff1a;Win10Visual Studio 2022 Community 在本次实验中需要用到第一篇文章实验内容的代码及环境&#xff0c;详情请见&#xff1a;传送门 目录 一、实验目的 二、实验步骤 1.Cohen-Sutherland算法 2.平移算法 3.旋转算法 一、实验目的 1.熟练掌握二维图形裁剪…

Java中的包装类

基本数据类型的豪华版---包装类基本数据类型包装类基本数据类型 在我们刚开始学习Java的时候,我们学习的应该就是Java中的八种基本数据类型: byte short int long float double char boolean 当时我们还说过Java是面向对象编程的语言,一切皆对象,但是受到当时知识的限制,我们还…

PHP MySQL 创建数据库

数据库存有一个或多个表。 你需要 CREATE 权限来创建或删除 MySQL 数据库。 使用 MySQLi 和 PDO 创建 MySQL 数据库 CREATE DATABASE 语句用于在 MySQL 中创建数据库。 在下面的实例中&#xff0c;创建了一个名为 "myDB" 的数据库&#xff1a; 实例 (MySQLi - 面…

系分 - 案例分析 - 架构设计(Web架构)

个人总结&#xff0c;仅供参考&#xff0c;欢迎加好友一起讨论 文章目录案例分析 - 系统设计&#xff08;Web架构&#xff09;Web架构知识点单台机器 到 数据库与Web服务器分离应用服务器集群负载均衡负载均衡技术静态与动态算法Session共享机制有状态与无状态持久化技术ORM数据…

22.Isaac教程--使用 Isaac Sight可视化

使用 Isaac Sight可视化 本节介绍如何使用 Isaac Sight 检查导航堆栈&#xff0c;机器人上当前发生的情况。 Sight 使用 Isaac 节点&#xff0c;该节点运行可在设备上连接的 Web 服务。 通过在 Chrome 网络浏览器中导航到 http://localhost:3000 来启动 Isaac Sight。 &#x…

(详解)java:无效的目标发行版: 11

目录 背景 解决方法 1.打开文件下的Project Structure 如下图&#xff1a; 2. 3. 4.完毕 背景 其实出现问题的真正原因是你下载别人的代码&#xff0c;但是别人的JDK属于java11 或者java9,但是你的为java8 或者更低 解决方法 1.打开文件下的Project Structure 如下图&a…

微信小程序——视图与逻辑,页面导航(导航到 tabBar 页面,导航到非 tabBar 页面)

一.页面导航1.什么是页面导航页面导航指的是页面之间的相互跳转。例如&#xff0c;浏览器中实现页面导航的方式有如下两种&#xff1a;a.< a &#xff1e;链接b.location . href2&#xff0e;小程序中实现页面导航的两种方式声明式导航在页面上声明一个&#xff1c; navigat…

nodejs框架koa,egg以及es6一起学

文章目录前言问题关于ES6-Generatoregg中的处理Promisenode的fsWorksheet Object参考文档前言 大家知道&#xff0c;eggjs框架是基于Koa开发的。关于koa&#xff0c;有一张经典的洋葱图&#xff1a; 这张图比较形象地展示了koa对于request和response的处理。每一层都留下扩展…

【RabbitMQ】高级篇,学习纪录+笔记

目录 一.高级特性 1.1消息的可靠投递 2.1Consumer Ack 3.1消费端限流 4.1TTL 5.1死信队列 6.1延迟队列 7.1日志与监控 7.1.1日志 7.1.2监控 8.1消息追踪 8.1.1Firehose 8.1.2rabbitmq_tracing 9.1消息可靠性保障&#xff08;思路&#xff09; 9.2消息幂等性保障&…

Hash算法,插入排序,希尔排序,选择排序,冒泡排序,归并排序,快速排序,堆排序,基数排序

Hash算法hash就是散列表&#xff0c;就是把任意长度的输入通过散列算法变换成固定长度的输出&#xff0c;该输出就是散列值。实质就是压缩映射&#xff0c;散列值的空间通常远小于输入的空间。常利用hash算法&#xff0c;将输入的一千万个检索串转化为题目中所说大约三百万个不…

《Linux Shell脚本攻略》学习笔记-第七章

7.1 简介 随着磁盘存储技术的发展&#xff0c;最简单地备份方法是添加新的磁盘设备或是使用云存储&#xff0c;而不再是依赖磁盘。 但是也需要压缩备份数据&#xff0c;降低存储空间需求以及传输时间。把数据存放在云端之前应该对其加密。数据在加密之前通常都要先进行归档和压…

2.2总线的性能指标

文章目录一、总线传输周期二、总线时钟周期三、总线工作频率四、总线时钟频率五、总线宽度六、总线带宽1.概念2.注意点3.案例分析4.总结补充七、总线复用八、信号线数九、总结这一节我们一起学习评价 总线性能的指标&#xff0c;这些指标很可能在选择题或者大题第一小题出现。一…

Git团队协作及分支策略

目录 分布式版本控制系统 访问模型 分支策略-Git flow feature 分支策略-Github flow 分支策略-Gitlab flow 主干开发模式 总结 分布式版本控制系统 分布式相比于集中式的最大区别在于开发者可以提交到本地&#xff0c;每个开发者通过克隆&#xff08;git clone&#…

Pytest参数化-下篇

&#x1f60e;&#x1f60e;原文出自&#xff1a;测个der&#xff0c;博主的公众号&#xff0c;格式美观一些。 关于参数化的其他案例 数据嵌套及多参数传参 import pytestpwd_datas [["QINGAN",{"user":"SHIER","pwd":"1234…

5个关键词回顾2022年个推技术实践

作为一家数据智能服务商&#xff0c;2022年每日互动&#xff08;个推&#xff09;在为开发者和行业客户提供优质服务的同时&#xff0c;不断砥砺创新&#xff0c;追逐技术前沿。个推还持续参与开发者生态建设&#xff0c;积极总结、分享自身技术实战经验&#xff0c;面向行业输…