JAVA字节流的两个重要子类FileOutputStream、FileInputStream

本文主要是介绍JAVA字节流的两个重要子类FileOutputStream、FileInputStream,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

字节流,字符集,字符流

Snipaste_2024-02-05_14-18-35.png
因为它们都是抽象类,所以要学习它们的子类


字节流的结构:Snipaste_2024-02-05_14-19-49.png

FileOutputStream

操作本地文件的字节输出流,可以把程序中的数据写入本地文件。

书写步骤:

  1. 创建字节输出流对象
  2. 写数据
  3. 释放资源

演示:

public class Test01 {public static void main(String[] args) throws IOException {//1.创建字节输出流对象/*细节:1.参数是字符串路径或是File对象都可以2.如果文件不存在,会创建一个新的文件,但要保证父级路径一定存在3.若文件已存在,构造方法,**会清空文件*/FileOutputStream fos = new FileOutputStream("..\\ioDemo\\a.txt");//2.写数据/*细节:write方法的参数是整数,但是实际上写到本地文件中的是整数在AScII上对应的字符如执意要写当前录入的数字,可以根据Ascii表的对应关系表示:48-0如:要写数字65fos.write(54);//6fos.write(53);//5*/fos.write(97)//afos.write(98)//b//3.释放资源:目的是解除资源的占用fos.close();}
}

image.pngimage.png

  1. 创建字节输出流 就好比程序 和 文件a.txt建立起一条道路
  2. 写数据 ** 就好比在这条路上运送货物,程序是始发地,文件是目的地**
  3. 释放资源 就是把这条路关闭

FileOutputStream写数据的三种方式:

image.png
第一种已经演示过了:
就演示第二种和第三种:

public class Test2 {public static void main(String[] args) throws IOException {//1创建字节输出流对象FileOutputStream fos = new FileOutputStream("..\\ioDemo\\a.txt");//2写入数据//字符串表示要写入的数据String s = "HelloWorld";//调用getBytes方法,得到字节数组byte[] bytes = s.getBytes();//放入字节数组fos.write(bytes);//3释放资源fos.close();}
}public class Test3 {public static void main(String[] args) throws IOException {FileOutputStream fos = new FileOutputStream("..\\ioDemo\\a.txt");byte[] bytes = {97, 98, 99};fos.write(bytes);fos.close();}
}

image.pngimage.png

public class Test2 {public static void main(String[] args) throws IOException {//1创建字节输出流对象FileOutputStream fos = new FileOutputStream("..\\ioDemo\\a.txt");//2写入数据//字符串表示要写入的数据String s = "HelloWorld";//调用getBytes方法,得到字节数组byte[] bytes = s.getBytes();//放入字节数组fos.write(bytes,3,3);//参数2表示开始索引,参数3表示长度//3释放资源fos.close();}
}

image.png


换行和续写

在上面的演示中有没有发现一个问题?

  1. 连续两次wirte写入的数据没有换行
    1. image.pngimage.png
  2. 每一次重启程序,之前写入的数据不能保留

解决:
换行:在连续write间再写一个换行符即可
image.png
换行演示:

public class Test02 {public static void main(String[] args) throws IOException {//创建字节流输出对象FileOutputStream fos=new FileOutputStream("..\\ioDemo\\a.txt");//写入第一个数据String str="JAVA";byte[] bytes = str.getBytes();fos.write(bytes);//换行回车String str2="\r\n";//表示回车、换行byte[] bytes1 = str2.getBytes();fos.write(bytes1);//写入第二个数据String str3="HelloWorld";byte[] bytes2 = str3.getBytes();fos.write(bytes2);fos.close();}
}

image.png

续写演示:
创建字节流对象时,在第二个参数打开续写开关即可,true
这样就不会清空之前的内容了,

public class Test02 {public static void main(String[] args) throws IOException {//创建字节流输出对象FileOutputStream fos=new FileOutputStream("..\\ioDemo\\a.txt",true);//换行回车String str2="\r\n";byte[] bytes1 = str2.getBytes();fos.write(bytes1);//写入第三个数据String str4="666";byte[] bytes3 = str4.getBytes();fos.write(bytes3);//释放资源fos.close();}
}

image.png在之前的基础上换行,添加666


FileInputStream

操作本地文件的字节输入流,可以把本地文件中的数据读取到程序中来。

书写步骤:

  1. 创建字节输入流对象
    1. 若要读取的文件不存在,就直接报错
  2. 读数据
    1. 无参的read方法一次读一个字节,读出来的是数据在ASCII上对应的数字
    2. 读到文件末尾了,read方法返回-1。
  3. 释放资源
    1. 每次使用完流必须要释放资源。

读取代码演示:

读取就像一个指针,读取一个,指针向后移动一格

public class t01 {public static void main(String[] args) throws IOException {//创建FileInputStream fis=new FileInputStream("..\\ioDemo\\a.txt");//读取数据,获取的是单个字符对应的ascii码,当然可以强转为字符int read1 = fis.read();System.out.println(read1);//97int read2 = fis.read();System.out.println((char) read2);// bint read3 = fis.read();System.out.println((char) read3);// cint read4 = fis.read();System.out.println(read4);//100int read5 = fis.read();System.out.println(read5);//101int read6 = fis.read();System.out.println(read6);//-1//释放资源fis.close();}
}

image.png


上面读数据的方法太慢了,当数据很大时,不断手动调用read方法肯定不合理

使用循环读取文件:

public class t02 {public static void main(String[] args) throws IOException {/**** 字节输入流的循环读取*///1.创建对象FileInputStream fis = new FileInputStream("..\\ioDemo\\a.txt");//2.循环读数据int len;//定义变量接收//读取到数据就进入while((len=fis.read())!=-1){System.out.print((char) len);}fis.close();//错误演示:错在多次调用了fis.read方法,是的指针发生了偏移/*while ((fis.read() != -1)) {int read = fis.read();System.out.println(read);}fis.close();*/}
}

image.png

控制台:
abcde


在上面我们只是读取了数据,
现在我们来试试,不但要读取数据,还要将数据输出到另一个位置,即__数据的拷贝
如图:
Snipaste_2024-02-05_19-00-14.png
那么我们现在就用无参的read方法来实现mp4数据的拷贝:

public class t3 {public static void main(String[] args) throws IOException {long l = System.currentTimeMillis();//开始时间//1.字节流输入对象--为了读取FileInputStream fis=new FileInputStream("E:\\aaa-FIle学习测试\\ccc\\movie.mp4");//2.字节流输出对象--为了写入FileOutputStream fos=new FileOutputStream("..\\ioDemo\\Copy.mp4");//**读取int b;//定义变量接b收bb//读取到数据就进入while((b=fis.read())!=-1){//**写入fos.write(b);//读出来的len是对应的ascii,但write方法会自动将数字转为对应字符}fos.close();fis.close();long l1 = System.currentTimeMillis();//结束时间long l2 = l1 - l;System.out.println("花费的毫秒值:"+l2);//花费时间}
}

控制台:
花费的毫秒值:12048

image.png
虽然说拷贝成功,但可以发现拷贝速度是很慢的。
这是因为无参的read方法实际上一次只能读取一个字节的数据。Snipaste_2024-02-05_20-50-52.png


现在我们就来学read的重载方法
image.png
形参是一个字节数组,好比是一个容器
这个字节数组的长度,就规定了每一次调用read方法读取到的数据的字节大小,(即一次读一个字节数组的数据,每次读取会尽可能把数组装满。)每次读取到的元素放入该字节数组
一般将字节数组的长度定义为1024的整数倍,如102410245(五兆)

在改写拷贝代码前,我们先熟悉这个重载方法:

public class t04 {public static void main(String[] args) throws IOException {//创建字节流输入对象FileInputStream fis=new FileInputStream("..\\ioDemo\\a.txt");//创建字节数组byte[]arr=new byte[2];//表示一次只读取两个字节,读取到的元素放入该数组//*开始读取int num1 = fis.read(arr);//返回值表示读取的字节个数System.out.println(num1);//***打印当前的字节数组String str1=new String(arr);System.out.println(str1);//读取int num2 = fis.read(arr);//返回值表示读取的字节个数System.out.println(num2);//***打印字节数组String str2=new String(arr);System.out.println(str2);//读取int num3 = fis.read(arr);//返回值表示读取的字节个数System.out.println(num3);//***此时只读取到最后一个元素e//***打印字节数组String str3=new String(arr);System.out.println(str3);//只读取到最后一个元素e,所以只能将cd中的c覆盖,所以打印ed//若要打印每次获取到的元素可以这样做:String str=new String (arr ,0 , num)}
}

image.png

2-------------------第一次获取了 两字节
ab------------------字节数组内的元素
2------------------第二次获取了 两字节
cd------------------字节数组内的元素
1------------------第二次获取了 一字节
ed------------------因为只读取到最后一个元素e,所以只能将之前cd的c覆盖,所以打印ed

//若要打印每次获取到的元素可以这样做:String str=new String (arr ,0 , num)
0表示开始索引,num表示当前读取到的字节个数

//一次读取一个字节数组FileInputStream fis=new FileInputStream("..\\ioDemo\\a.txt");byte[]arr=new byte[2];//表示一次只读取两个字节,读取到的元素放入该数组//读取int num1 = fis.read(arr);//返回值表示读取的字节个数System.out.println(num1);//***打印字节数组String str1=new String(arr,0,num1);System.out.println(str1);//读取int num2 = fis.read(arr);//返回值表示读取的字节个数System.out.println(num2);//***打印字节数组String str2=new String(arr,0,num2);System.out.println(str2);//读取int num3 = fis.read(arr);//返回值表示读取的字节个数System.out.println(num3);//***此时只读取到最后一个元素e//***打印字节数组String str3=new String(arr,0,num3);System.out.println(str3);

2
ab
2
cd
1
e


正式改进拷贝代码:

public class t5 {public static void main(String[] args) throws IOException {/**** 文件拷贝改写*/long start = System.currentTimeMillis();//开始时间//创建字节流输入,输出对象FileInputStream fis=new FileInputStream("E:\\aaa-FIle学习测试\\ccc\\movie.mp4");FileOutputStream fos=new FileOutputStream("..\\ioDemo\\Copy.mp4");byte[]bytes=new byte[1024*1024*5];//一次读取五兆,即表示写入速度,又表示存储位置int len;//每次的写入字节大小while((len=fis.read(bytes))!=-1){fos.write(bytes,0,len);//再将每次写入的数据,输出到目的地}//先调用者,后关闭fos.close();fis.close();long end = System.currentTimeMillis();//结束时间System.out.println((end-start));//花费时间}
}

225

速度明显变快了。
image.png

这篇关于JAVA字节流的两个重要子类FileOutputStream、FileInputStream的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Spring Cloud Hystrix原理与注意事项小结

《SpringCloudHystrix原理与注意事项小结》本文介绍了Hystrix的基本概念、工作原理以及其在实际开发中的应用方式,通过对Hystrix的深入学习,开发者可以在分布式系统中实现精细... 目录一、Spring Cloud Hystrix概述和设计目标(一)Spring Cloud Hystr

Spring Boot整合消息队列RabbitMQ的实现示例

《SpringBoot整合消息队列RabbitMQ的实现示例》本文主要介绍了SpringBoot整合消息队列RabbitMQ的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的... 目录RabbitMQ 简介与安装1. RabbitMQ 简介2. RabbitMQ 安装Spring

springMVC返回Http响应的实现

《springMVC返回Http响应的实现》本文主要介绍了在SpringBoot中使用@Controller、@ResponseBody和@RestController注解进行HTTP响应返回的方法,... 目录一、返回页面二、@Controller和@ResponseBody与RestController

JAVA集成本地部署的DeepSeek的图文教程

《JAVA集成本地部署的DeepSeek的图文教程》本文主要介绍了JAVA集成本地部署的DeepSeek的图文教程,包含配置环境变量及下载DeepSeek-R1模型并启动,具有一定的参考价值,感兴趣的... 目录一、下载部署DeepSeek1.下载ollama2.下载DeepSeek-R1模型并启动 二、J

springboot rocketmq配置生产者和消息者的步骤

《springbootrocketmq配置生产者和消息者的步骤》本文介绍了如何在SpringBoot中集成RocketMQ,包括添加依赖、配置application.yml、创建生产者和消费者,并展... 目录1. 添加依赖2. 配置application.yml3. 创建生产者4. 创建消费者5. 使用在

Spring Retry 实现乐观锁重试实践记录

《SpringRetry实现乐观锁重试实践记录》本文介绍了在秒杀商品SKU表中使用乐观锁和MybatisPlus配置乐观锁的方法,并分析了测试环境和生产环境的隔离级别对乐观锁的影响,通过简单验证,... 目录一、场景分析 二、简单验证 2.1、可重复读 2.2、读已提交 三、最佳实践 3.1、配置重试模板

Spring中@Lazy注解的使用技巧与实例解析

《Spring中@Lazy注解的使用技巧与实例解析》@Lazy注解在Spring框架中用于延迟Bean的初始化,优化应用启动性能,它不仅适用于@Bean和@Component,还可以用于注入点,通过将... 目录一、@Lazy注解的作用(一)延迟Bean的初始化(二)与@Autowired结合使用二、实例解

SpringBoot使用Jasypt对YML文件配置内容加密的方法(数据库密码加密)

《SpringBoot使用Jasypt对YML文件配置内容加密的方法(数据库密码加密)》本文介绍了如何在SpringBoot项目中使用Jasypt对application.yml文件中的敏感信息(如数... 目录SpringBoot使用Jasypt对YML文件配置内容进行加密(例:数据库密码加密)前言一、J

Java中有什么工具可以进行代码反编译详解

《Java中有什么工具可以进行代码反编译详解》:本文主要介绍Java中有什么工具可以进行代码反编译的相关资,料,包括JD-GUI、CFR、Procyon、Fernflower、Javap、Byte... 目录1.JD-GUI2.CFR3.Procyon Decompiler4.Fernflower5.Jav

Spring Boot 中正确地在异步线程中使用 HttpServletRequest的方法

《SpringBoot中正确地在异步线程中使用HttpServletRequest的方法》文章讨论了在SpringBoot中如何在异步线程中正确使用HttpServletRequest的问题,... 目录前言一、问题的来源:为什么异步线程中无法访问 HttpServletRequest?1. 请求上下文与线