IO流上:概述、字符流、缓冲区
一、IO流概述
概述:
IO流简单来说就是Input和Output流,IO流主要是用来处理设备之间的数据传输,java对于数据的操作都是通过流实现,而java用于操作流的对象都在IO包中。
分类:
按操作数据分为:字节流和字符流。 如:Reader和InpurStream
按流向分:输入流和输出流。如:InputStream和OutputStream
IO流常用的基类:
* InputStream , OutputStream
字符流的抽象基类:
* Reader , Writer
由上面四个类派生的子类名称都是以其父类名作为子类的后缀:
如:FileReader和FileInputStream
二、字符流
1. 字符流简介:
* 字符流中的对象融合了编码表,也就是系统默认的编码表。我们的系统一般都是GBK编码。
* 字符流只用来处理文本数据,字节流用来处理媒体数据。
* 数据最常见的表现方式是文件,字符流用于操作文件的子类一般是FileReader和FileWriter。
2.字符流读写:
注意事项:
* 写入文件后必须要用flush()刷新。
* 用完流后记得要关闭流
* 使用流对象要抛出IO异常
*定义文件路径时,可以用“/”或者“\\”。
*在创建一个文件时,如果目录下有同名文件将被覆盖。
*在读取文件时,必须保证该文件已存在,否则出异常示例1:在硬盘上创建一个文件,并写入一些文字数据
- classFireWriterDemo{
- publicstaticvoidmain(String[]args)throwsIOException{//需要对IO异常进行处理
- //创建一个FileWriter对象,该对象一被初始化就必须要明确被操作的文件。
- //而且该文件会被创建到指定目录下。如果该目录有同名文件,那么该文件将被覆盖。
- FileWriterfw=newFileWriter("F:\\1.txt");//目的是明确数据要存放的目的地。
- //调用write的方法将字符串写到流中
- fw.write("helloworld!");
- //刷新流对象缓冲中的数据,将数据刷到目的地中
- fw.flush();
- //关闭流资源,但是关闭之前会刷新一次内部缓冲中的数据。当我们结束输入时候,必须close();
- fw.write("first_test");
- fw.close();
- //flush和close的区别:flush刷新后可以继续输入,close刷新后不能继续输入。
- }
- }
示例2:FileReader的reade()方法.
要求:用单个字符和字符数组进行分别读取
- classFileReaderDemo{
- publicstaticvoidmain(String[]args){
- characters();
- }
- /*****************字符数组进行读取*********************/
- privatestaticvoidcharacters(){
- try{
- FileReaderfr=newFileReader("Demo.txt");
- char[]buf=newchar[6];
- //将Denmo中的文件读取到buf数组中。
- intnum=0;
- while((num=fr.read(buf))!=-1){
- //String(char[]value,intoffest,intcount)分配一个新的String,包含从offest开始的count个字符
- sop(newString(buf,0,num));
- }
- sop('\n');
- fr.close();
- }
- catch(IOExceptione){
- sop(e.toString());
- }
- }
- /*****************单个字母读取*************************/
- privatestaticvoidsingleReader(){
- try{
- //创建一个文件读取流对象,和指定名称的文件关联。
- //要保证文件已经存在,否则会发生异常:FileNotFoundException
- FileReaderfr=newFileReader("Demo.txt");
- //如何调用读取流对象的read方法?
- //read()方法,一次读取一个字符,并且自动往下读。如果到达末尾则返回-1
- intch=0;
- while((ch=fr.read())!=-1){
- sop((char)ch);
- }
- sop('\n');
- fr.close();
- /*intch=fr.read();
- sop("ch="+(char)ch);
- intch2=fr.read();
- sop("ch2="+(char)ch2);
- //使用结束注意关闭流
- fr.close();*/
- }
- catch(IOExceptione){
- sop(e.toString());
- }
- }
- /**********************Println************************/
- privatestaticvoidsop(Objectobj){
- System.out.print(obj);
- }
- }
示例3:对已有文件的数据进行续写
- importjava.io.*;
- classFileWriterDemo3{
- publicstaticvoidmain(String[]args){
- try{
- //传递一个参数,代表不覆盖已有的数据。并在已有数据的末尾进行数据续写
- FileWriterfw=newFileWriter("F:\\java_Demo\\day9_24\\demo.txt",true);
- fw.write("ischaractortable?");
- fw.close();
- }
- catch(IOExceptione){
- sop(e.toString());
- }
- }
- /**********************Println************************/
- privatestaticvoidsop(Objectobj)
- {
- System.out.println(obj);
- }
- }
练习:
将F盘的一个文件复制到E盘。
思考:
其实就是将F盘下的文件数据存储到D盘的一个文件中。
步骤:
1.在D盘创建一个文件,存储F盘中文件的数据。
2.定义读取流和F:盘文件关联。
3.通过不断读写完成数据存储。
4.关闭资源。源码:
- importjava.io.*;
- importjava.util.Scanner;
- classCopyText{
- publicstaticvoidmain(String[]args)throwsIOException{
- sop("请输入要拷贝的文件的路径:");
- Scannerin=newScanner(System.in);
- Stringsource=in.next();
- sop("请输入需要拷贝到那个位置的路径以及生成的文件名:");
- Stringdestination=in.next();
- in.close();
- CopyTextDemo(source,destination);
- }
- /*****************文件Copy*********************/
- privatestaticvoidCopyTextDemo(Stringsource,Stringdestination){
- try{
- FileWriterfw=newFileWriter(destination);
- FileReaderfr=newFileReader(source);
- char[]buf=newchar[1024];
- //将Denmo中的文件读取到buf数组中。
- intnum=0;
- while((num=fr.read(buf))!=-1){
- //String(char[]value,intoffest,intcount)分配一个新的String,包含从offest开始的count个字符
- fw.write(newString(buf,0,num));
- }
- fr.close();
- fw.close();
- }
- catch(IOExceptione){
- sop(e.toString());
- }
- }
- /**********************Println************************/
- privatestaticvoidsop(Objectobj){
- System.out.println(obj);
- }
- }
三、缓冲区
1. 字符流的缓冲区:BufferedReader和BufferedWreiter
* 缓冲区的出现时为了提高流的操作效率而出现的.
* 需要被提高效率的流作为参数传递给缓冲区的构造函数
* 在缓冲区中封装了一个数组,存入数据后一次取出
BufferedReader示例:
读取流缓冲区提供了一个一次读一行的方法readline,方便对文本数据的获取。
readline()只返回回车符前面的字符,不返回回车符。如果是复制的话,必须加入newLine(),写入回车符newLine()是java提供的多平台换行符写入方法。
- importjava.io.*;
- classBufferedReaderDemo{
- publicstaticvoidmain(String[]args)throwsIOException{
- //创建一个字符读取流流对象,和文件关联
- FileReaderrw=newFileReader("buf.txt");
- //只要将需要被提高效率的流作为参数传递给缓冲区的构造函数即可
- BufferedReaderbrw=newBufferedReader(rw);
- for(;;){
- Strings=brw.readLine();
- if(s==null)break;
- System.out.println(s);
- }
- brw.close();//关闭输入流对象
- }
- }
BufferedWriter示例:
- importjava.io.*;
- classBufferedWriterDemo{
- publicstaticvoidmain(String[]args)throwsIOException{
- //创建一个字符写入流对象
- FileWriterfw=newFileWriter("buf.txt");
- //为了提高字符写入效率,加入了缓冲技术。
- //只要将需要被提高效率的流作为参数传递给缓冲区的构造函数即可
- BufferedWriterbfw=newBufferedWriter(fw);
- //bfw.write("abc\r\nde");
- //bfw.newLine();这行代码等价于bfw.write("\r\n"),相当于一个跨平台的换行符
- //用到缓冲区就必须要刷新
- for(intx=1;x<5;x++){
- bfw.write("abc");
- bfw.newLine();//java提供了一个跨平台的换行符newLine();
- bfw.flush();
- }
- bfw.flush();//刷新缓冲区
- bfw.close();//关闭缓冲区,但是必须要先刷新
- //注意,关闭缓冲区就是在关闭缓冲中的流对象
- fw.close();//关闭输入流对象
- }
- }
2.装饰设计模式
装饰设计模式::::
要求:自定义一些Reader类,读取不同的数据(装饰和继承的区别)
MyReader //专门用于读取数据的类
|--MyTextReader
|--MyBufferTextReader
|--MyMediaReader
|--MyBufferMediaReader
|--MyDataReader
|--MyBufferDataReader
如果将他们抽取出来,设计一个MyBufferReader,可以根据传入的类型进行增强
class MyBufferReader {
MyBufferReader (MyTextReader text) {}
MyBufferReader (MyMediaReader media) {}
MyBufferReader (MyDataReader data) {}
}
但是上面的类拓展性很差。找到其参数的共同类型,通过多态的形式,可以提高拓展性
class MyBufferReader extends MyReader{
private MyReader r; //从继承变为了组成模式 装饰设计模式
MyBufferReader(MyReader r) {}
}
优化后的体系:
|--MyTextReader
|--MyMediaReader
|--MyDataReader
|--MyBufferReader //增强上面三个。装饰模式比继承灵活,
避免继承体系的臃肿。降低类与类之间的耦合性
装饰类只能增强已有的对象,具备的功能是相同的。所以装饰类和被装饰类属于同一个体系MyBuffereReader类: 自己写一个MyBuffereReader类,功能与BuffereReader相同
- classMyBufferedReader1extendsReader{
- privateReaderr;
- MyBufferedReader1(Readerr){
- this.r=r;
- }
- //一次读一行数据的方法
- publicStringmyReaderline()throwsIOException{
- //定义一个临时容器,原BufferReader封装的是字符数组。
- //为了演示方便。定义一个StringBuilder容器。最终要将数据变成字符串
- StringBuildersb=newStringBuilder();
- intch=0;
- while((ch=r.read())!=-1)
- {
- if(ch=='\r')
- continue;
- if(ch=='\n')//遇到换行符\n,返回字符串
- returnsb.toString();
- else
- sb.append((char)ch);
- }
- if(sb.length()!=0)//当最后一行不是以\n结束时候,这里需要判断
- returnsb.toString();
- returnnull;
- }
- /*
- 需要覆盖Reader中的抽象方法close(),read();
- */
- publicvoidclose()throwsIOException{
- r.close();
- }
- publicintread(char[]cbuf,intoff,intlen)throwsIOException{//覆盖read方法
- returnr.read(cbuf,off,len);
- }
- publicvoidmyClose()throwsIOException{
- r.close();
- }
- }