字节流和字符流详解-程序员宅基地

技术标签: java  IO  学习笔记  

1.流的概念

  • 在编程中是一种抽象的概念,就好比“水流”,从一段流向另一端
  • 在程序中所有的数据都是以流的方式进行传输或保存的,程序需要数据的时候要使用输入流读取数据,而当程序需要将一些数据保存起来的时候,就要使用输出流完成。
  • 程序中的输入输出都是以流的形式保存的,流中保存的实际上全都是字节文件。

2.流的分类

按照流向可以分为:输入流(如:键盘,麦克风),输出流(如:显示器,音箱)
按照传输单位可以分为:字节流和字符流

3.什么是字节流,什么是字符流

字节流: 它处理单元为1个字节(byte),操作字节和字节数组,存储的是二进制文件,如果是音频文件、图片、歌曲,就用字节流好点(1byte = 8位);

字符流: 它处理的单元为2个字节的Unicode字符,分别操作字符、字符数组或字符串,字符流是由Java虚拟机将字节转化为2个字节的Unicode字符为单位的字符而成的,如果是关系到中文(文本)的,用字符流好点(1Unicode = 2字节 = 16位);

所有文件的储存是都是字节(byte)的储存,在磁盘上保留的并不是文件的字符而是先把字符编码成字节,再储存这些字节到磁盘。在读取文件(特别是文本文件)时,也是一个字节一个字节地读取以形成字节序列。

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

字节流是最基本的,所有的InputStrem和OutputStream的子类都是,主要用在处理二进制数据,它是按字节来处理的 但实际中很多的数据是文本,又提出了字符流的概念,它是按虚拟机的encode来处理,也就是要进行字符集的转化 这两个之间通过 InputStreamReader,OutputStreamWriter来关联,实际上是通过byte[]和String来关联 在实际开发中出现的汉字问题实际上都是在字符流和字节流之间转化不统一而造成的

最简单的区分字节流和字符流

万物皆文件,那就将文件在记事本里面打开,如果打开后能看的懂的就是字符流,如果看不懂那就是字节流

问题:
1.word.doc 数据字节流还是字符流?
答:.doc数据字节流。
2.Excel 数据字节流还是字符流?
答:要根据保存的格式进行判断,如果是保存为.csv那么他就是字符流,如果是其他的则数据字节流。

字符和字节操作

在这里插入图片描述
如何简单记住字节流和字符流

  • 字符流是以Reader,Writer结尾的
  • 字符节流是以InputStream或OutputStream结尾的

4.字节流的使用

字节流主要是操作byte类型数据,以byte数组为准,主要操作类就是OutputStream、InputStream。

字节输出流:OutputStream,OutputStream是整个IO包中字节输出流的最大父类

Closeable表示可以关闭的操作

示例为:图片的复制

步骤:
1.读入二进制文件(输入流,FileInputStream)
2.写入二进制文件(输出流,FileOutputStream)

前请了解:try-with-resource,这是JDK1.7提供的新方法,让代码的可读性更高,但是并不是所有的对象都可以这样,当调用的类实现了closable接口就可以使用此种方式

4.1.普通字节流的操作
代码案例
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;

/**
 * 复制D:\io_test下的加菲猫文件到D:\io_test这个目录下
 */

public class IoDemo4 {
    

    public static void main(String[] args) throws IOException {
    
        //输入文件目录
        String srcFilePath = "D:\\io_test\\加菲猫.jpg";
        //输出文件目录(拷贝的路径)
        String destFilePath = "D:\\io_test\\加菲猫2.jpg";
        FileInputStream fileInputStream = null;
        FileOutputStream fileOutputStream = null;
        try {
    
            //输入流,传入路径
            fileInputStream = new FileInputStream(srcFilePath);
            //输出流
             fileOutputStream = new FileOutputStream(destFilePath);
            byte[] bytes = new byte[1024];
            try {
    
                int count = 0;
                //如果不等于-1说明还没有读取完成,要继续读取
                while((count = fileInputStream.read(bytes)) != -1){
    
                    //还有内容要读取,然后从0开始读取,上面count有读了多少个,那么就写多少个
                    fileOutputStream.write(bytes,0,count);
                }
            } catch (IOException e) {
    
                e.printStackTrace();
            }

        } catch (FileNotFoundException e) {
    
            e.printStackTrace();
        }finally {
    
            //用完之后一定要关闭流,但是得判断不是空的时候才需要去关闭流
            if(fileInputStream != null){
    
                fileInputStream.close();
            }
            if(fileOutputStream != null){
    
                fileOutputStream.close();
            }
        }
    }
}

执行结果
在这里插入图片描述

注意:
1.因为字节流主要操作byte类型数据,注意里面参数的含义,如果是读取的话,比较常用的是第二种
在这里插入图片描述

(FileOutputStream)也是一样,读取的时候一般用第一个
在这里插入图片描述

2.读取数据的返回结果(FileInputStream)
在这里插入图片描述

3.最后一定要关闭流

  • 因为每天计算机打开的文件的数量是有限的,以liunx为例,最大可以打开66533个文件,
  • 但是无论执行结果怎么样他都要关闭流,那么就需要将他放在finally里面,所有上面的声明也得放在外面
  • 如果传入的地址和目标地址都是错的,也就是找不到,如果传入的地址错了,那么还没有初始化就已经被调用关闭流了,所以要进行判断
4.2.带有缓冲区的字节流的操作

什么是缓存区?
定义:缓存区相当于缓存,它是存在内存中的
写操作:
没有使用缓存区:CPU读取每个字节之后直接操作磁盘(性能比较底)进行写完,写操作的瓶颈就会出现,因为每个字节都会操作一次磁盘
使用缓冲区:那么每次会将字符放入缓存区(内存),等缓冲区满了之后,才一次性写入磁盘
因为内存的操作速度远远大于磁盘,因此带缓冲区的输入流和输出流实现的效率就非常高(比如扔垃圾,一次性扔完和一次次扔肯定消耗的时间是有很大差距的)

案例实现:复制D:\io_test\下的加菲猫—>D:\io_test\下变成加菲猫3

代码实现
import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.FileInputStream;
import java.io.FileOutputStream;

/**
 * 带有缓冲区的字节流操作:图片复制
 *
 */
public class IoDemo5 {
    
    public static void main(String[] args) {
    
        //输入文件目录
        String srcFilePath = "D:\\io_test\\加菲猫.jpg";
        //输出文件目录(拷贝的路径)
        String destFilePath = "D:\\io_test\\加菲猫3.jpg";
        try {
    
            //因为带有缓存区的是基于原始的类进行操作的
            BufferedInputStream bufferedInputStream =
                    new BufferedInputStream(new FileInputStream(srcFilePath));
            BufferedOutputStream bufferedOutputStream =
                    new BufferedOutputStream(new FileOutputStream(destFilePath));
            {
    
                        byte[] bytes = new byte[1024];
                        int count = 0;
                        while ((count = bufferedInputStream.read(bytes)) != -1){
    
                            bufferedOutputStream.write(bytes,0,count);
                        }
            }
        }catch (Exception e){
    
            e.printStackTrace();
        }
    }
}

执行结果
在这里插入图片描述

5.字符流的使用

5.1.普通的字符流操作

在程序中一个字符等于两个字节,那么java提供了Reader、Writer两个专门操作字符流的类。

字符输出流:Writer。
字符输入流:Reader

示例一:内容的写入:D:\io_test\1下创建一个text.txt文件,并写入内容“欢迎来到Java~”
import java.io.FileWriter;
import java.io.IOException;

/**
 * 内容的写入:D:\io_test\1下创建一个text.txt文件,并写入内容
 */
public class IoDemo7 {
    
    public static void main(String[] args) throws IOException {
    
        //定义文件地址
        String filePath = "D:\\io_test\\1\\text.txt";
        //写入内容
        String content = "欢迎来到Java~";
        //因为是写入文件,所以要使用FileWriter方法
        FileWriter fileWriter = null;
        try {
    
            fileWriter = new FileWriter(filePath);
            fileWriter.write(content);
        } catch (IOException e) {
    
            e.printStackTrace();
        }finally {
    
        //关闭流
            fileWriter.close();
        }
    }
}

**注意:**如果不关闭流,虽然也能创建文件成功,不报错,但是可能存在内容无法写入的情况

示例二:但是上面这种操作如果是对于多行输入的话那么就回被覆盖
因为这里的参数append会被默认成false,所以如果是多行写入的话
需要手动设置成true

在这里插入图片描述
来看看在刚才那种被覆盖掉的情况
在这里插入图片描述

代码实现:那么来看看将append设置成true
public class IoDemo7 {
    
    public static void main(String[] args) throws IOException {
    
        //定义文件地址
        String filePath = "D:\\io_test\\1\\text.txt";
        //写入内容
        String content = "欢迎来到java~,";
        //因为是写入文件,所以要使用FileWriter方法
        FileWriter fileWriter = null;
        try {
    
            fileWriter = new FileWriter(filePath,true);
            fileWriter.write(content);
        } catch (IOException e) {
    
            e.printStackTrace();
        }finally {
    
            fileWriter.close();
        }
    }
}

执行结果
在这里插入图片描述

示例三:文件的读取,比如将文件下的内容读取到控制台

文件下的内容

import java.io.File;
import java.io.FileNotFoundException;
import java.util.Scanner;

/**
 * 读取文件内容
 */
public class IoDemo8 {
    
    public static void main(String[] args) throws FileNotFoundException {
    
        String filePath = "D:\\io_test\\1\\tt.txt";
        Scanner scanner = new Scanner(new File(filePath));
        //因为读取的内容可能不止一条,所以加上while循环
        while (scanner.hasNext()){
    
            System.out.println(scanner.nextLine());
        }
    }
}

执行效果
在这里插入图片描述

 示例四:将内容读取到内存中,如:将一个路径下的内容复制到另一个路径下

代码实现

import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;

/**
 *将一个文件的内容写入到另一个文件下
 * 如:D:\io_test\1\text.txt -> text2.txt
 */
public class IoDemo9 {
    
    public static void main(String[] args) throws IOException {
    
        //读取文件的地址
        String srcFilePath = "D:\\io_test\\1\\text.txt";
        //目标文件的地址
        String destFilePath = "D:\\io_test\\1\\text2.txt";
        FileReader fileReader = null;
        FileWriter fileWriter = null;
        //先读后写
        try{
    
             fileReader = new FileReader(srcFilePath);
             fileWriter = new FileWriter(destFilePath);
            {
    
                //读操作
                char[] chars = new char[1024];
                while (true){
    
                    int count = fileReader.read(chars);
                    //判断是否等于-1,如果等于-1,说明已经读取完了,跳出循环
                    if(count == -1){
    
                        break;
                    }
                    //写操作
                    fileWriter.write(chars,0,count);
                }
            }
        }catch (Exception e){
    
            e.printStackTrace();
        }finally {
    
            fileReader.close();
            fileWriter.close();
        }
    }
}

执行结果
在这里插入图片描述

5.2.带有缓冲区的字符流操作
示例:如上述的示例四

代码实现

import java.io.*;

/**
 * 带有缓冲区的字符流操作
 * 如:内容的复制
 */
public class IoDemo10 {
    
    public static void main(String[] args) throws IOException {
    
        String srcFilePath = "D:\\io_test\\1\\tt.txt";
        String destFilePath = "D:\\io_test\\1\\tt2.txt";
        BufferedReader bufferedReader = null;
        BufferedWriter bufferedWriter = null;
        //先读取,后写入
        try {
    
             bufferedReader = new BufferedReader(new FileReader(srcFilePath));
             bufferedWriter = new BufferedWriter(new FileWriter(destFilePath));
            {
    
                char[] chars = new char[1024];
                while (true){
    
                    int count = bufferedReader.read(chars);
                    if(count == -1){
    
                        break;
                    }
                    bufferedWriter.write(chars,0,count);
                }
            }
        }catch (Exception e){
    
            e.printStackTrace();
        }finally {
    
            bufferedReader.close();
            bufferedWriter.close();
        }
    }
}

执行结果
在这里插入图片描述

6.总结对比字节流和字符流

  • 字节流操作的基本单元是字节;字符流操作的基本单元为Unicode码元。

  • 字节流在操作的时候本身不会用到缓冲区的,是与文件本身直接操作的;而字符流在操作的时候使用到缓冲区的。

  • 所有文件的存储都是字节(byte)的存储,在磁盘上保留的是字节。

  • 在使用字节流操作中,即使没有关闭资源(close方法),也能输出;而字符流不使用close方法的话,不会输出任何内容

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/wkh18891843165/article/details/115310276

智能推荐

什么是内部类?成员内部类、静态内部类、局部内部类和匿名内部类的区别及作用?_成员内部类和局部内部类的区别-程序员宅基地

文章浏览阅读3.4k次,点赞8次,收藏42次。一、什么是内部类?or 内部类的概念内部类是定义在另一个类中的类;下面类TestB是类TestA的内部类。即内部类对象引用了实例化该内部对象的外围类对象。public class TestA{ class TestB {}}二、 为什么需要内部类?or 内部类有什么作用?1、 内部类方法可以访问该类定义所在的作用域中的数据,包括私有数据。2、内部类可以对同一个包中的其他类隐藏起来。3、 当想要定义一个回调函数且不想编写大量代码时,使用匿名内部类比较便捷。三、 内部类的分类成员内部_成员内部类和局部内部类的区别

分布式系统_分布式系统运维工具-程序员宅基地

文章浏览阅读118次。分布式系统要求拆分分布式思想的实质搭配要求分布式系统要求按照某些特定的规则将项目进行拆分。如果将一个项目的所有模板功能都写到一起,当某个模块出现问题时将直接导致整个服务器出现问题。拆分按照业务拆分为不同的服务器,有效的降低系统架构的耦合性在业务拆分的基础上可按照代码层级进行拆分(view、controller、service、pojo)分布式思想的实质分布式思想的实质是为了系统的..._分布式系统运维工具

用Exce分析l数据极简入门_exce l趋势分析数据量-程序员宅基地

文章浏览阅读174次。1.数据源准备2.数据处理step1:数据表处理应用函数:①VLOOKUP函数; ② CONCATENATE函数终表:step2:数据透视表统计分析(1) 透视表汇总不同渠道用户数, 金额(2)透视表汇总不同日期购买用户数,金额(3)透视表汇总不同用户购买订单数,金额step3:讲第二步结果可视化, 比如, 柱形图(1)不同渠道用户数, 金额(2)不同日期..._exce l趋势分析数据量

宁盾堡垒机双因素认证方案_horizon宁盾双因素配置-程序员宅基地

文章浏览阅读3.3k次。堡垒机可以为企业实现服务器、网络设备、数据库、安全设备等的集中管控和安全可靠运行,帮助IT运维人员提高工作效率。通俗来说,就是用来控制哪些人可以登录哪些资产(事先防范和事中控制),以及录像记录登录资产后做了什么事情(事后溯源)。由于堡垒机内部保存着企业所有的设备资产和权限关系,是企业内部信息安全的重要一环。但目前出现的以下问题产生了很大安全隐患:密码设置过于简单,容易被暴力破解;为方便记忆,设置统一的密码,一旦单点被破,极易引发全面危机。在单一的静态密码验证机制下,登录密码是堡垒机安全的唯一_horizon宁盾双因素配置

谷歌浏览器安装(Win、Linux、离线安装)_chrome linux debian离线安装依赖-程序员宅基地

文章浏览阅读7.7k次,点赞4次,收藏16次。Chrome作为一款挺不错的浏览器,其有着诸多的优良特性,并且支持跨平台。其支持(Windows、Linux、Mac OS X、BSD、Android),在绝大多数情况下,其的安装都很简单,但有时会由于网络原因,无法安装,所以在这里总结下Chrome的安装。Windows下的安装:在线安装:离线安装:Linux下的安装:在线安装:离线安装:..._chrome linux debian离线安装依赖

烤仔TVの尚书房 | 逃离北上广?不如押宝越南“北上广”-程序员宅基地

文章浏览阅读153次。中国发达城市榜单每天都在刷新,但无非是北上广轮流坐庄。北京拥有最顶尖的文化资源,上海是“摩登”的国际化大都市,广州是活力四射的千年商都。GDP和发展潜力是衡量城市的数字指...

随便推点

java spark的使用和配置_使用java调用spark注册进去的程序-程序员宅基地

文章浏览阅读3.3k次。前言spark在java使用比较少,多是scala的用法,我这里介绍一下我在项目中使用的代码配置详细算法的使用请点击我主页列表查看版本jar版本说明spark3.0.1scala2.12这个版本注意和spark版本对应,只是为了引jar包springboot版本2.3.2.RELEASEmaven<!-- spark --> <dependency> <gro_使用java调用spark注册进去的程序

汽车零部件开发工具巨头V公司全套bootloader中UDS协议栈源代码,自己完成底层外设驱动开发后,集成即可使用_uds协议栈 源代码-程序员宅基地

文章浏览阅读4.8k次。汽车零部件开发工具巨头V公司全套bootloader中UDS协议栈源代码,自己完成底层外设驱动开发后,集成即可使用,代码精简高效,大厂出品有量产保证。:139800617636213023darcy169_uds协议栈 源代码

AUTOSAR基础篇之OS(下)_autosar 定义了 5 种多核支持类型-程序员宅基地

文章浏览阅读4.6k次,点赞20次,收藏148次。AUTOSAR基础篇之OS(下)前言首先,请问大家几个小小的问题,你清楚:你知道多核OS在什么场景下使用吗?多核系统OS又是如何协同启动或者关闭的呢?AUTOSAR OS存在哪些功能安全等方面的要求呢?多核OS之间的启动关闭与单核相比又存在哪些异同呢?。。。。。。今天,我们来一起探索并回答这些问题。为了便于大家理解,以下是本文的主题大纲:[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-JCXrdI0k-1636287756923)(https://gite_autosar 定义了 5 种多核支持类型

VS报错无法打开自己写的头文件_vs2013打不开自己定义的头文件-程序员宅基地

文章浏览阅读2.2k次,点赞6次,收藏14次。原因:自己写的头文件没有被加入到方案的包含目录中去,无法被检索到,也就无法打开。将自己写的头文件都放入header files。然后在VS界面上,右键方案名,点击属性。将自己头文件夹的目录添加进去。_vs2013打不开自己定义的头文件

【Redis】Redis基础命令集详解_redis命令-程序员宅基地

文章浏览阅读3.3w次,点赞80次,收藏342次。此时,可以将系统中所有用户的 Session 数据全部保存到 Redis 中,用户在提交新的请求后,系统先从Redis 中查找相应的Session 数据,如果存在,则再进行相关操作,否则跳转到登录页面。此时,可以将系统中所有用户的 Session 数据全部保存到 Redis 中,用户在提交新的请求后,系统先从Redis 中查找相应的Session 数据,如果存在,则再进行相关操作,否则跳转到登录页面。当数据量很大时,count 的数量的指定可能会不起作用,Redis 会自动调整每次的遍历数目。_redis命令

URP渲染管线简介-程序员宅基地

文章浏览阅读449次,点赞3次,收藏3次。URP的设计目标是在保持高性能的同时,提供更多的渲染功能和自定义选项。与普通项目相比,会多出Presets文件夹,里面包含着一些设置,包括本色,声音,法线,贴图等设置。全局只有主光源和附加光源,主光源只支持平行光,附加光源数量有限制,主光源和附加光源在一次Pass中可以一起着色。URP:全局只有主光源和附加光源,主光源只支持平行光,附加光源数量有限制,一次Pass可以计算多个光源。可编程渲染管线:渲染策略是可以供程序员定制的,可以定制的有:光照计算和光源,深度测试,摄像机光照烘焙,后期处理策略等等。_urp渲染管线