(22)I/O流-字节流/字符流

一,字节输出流OutputStream

OutputStream此抽象类,是表示输出字节流的所有类的超类。操作的数据都是字节,定义了输出字节流的基本共性功能方法。输出流中定义都是写write方法,如下图:

1.1FileOutputStream类

OutputStream有很多子类,其中子类FileOutputStream可用来写入数据到文件。

FileOutputStream类,即文件输出流,是用于将数据写入File的输出流。

FileOutputStream类写入数据到文件中

将数据写到文件中,代码演示:

public class FileOutputStreamDemo {

    public static void main(String[] args) throws IOException {

        //需求:将数据写入到文件中。

        //创建存储数据的文件。

        Filefile =new File("c:\\file.txt");

        //创建一个用于操作文件的字节输出流对象。一创建就必须明确数据存储目的地。

        //输出流目的是文件,会自动创建。如果文件存在,则覆盖。

        FileOutputStreamfos =new FileOutputStream(file);

        //调用父类中的write方法。

        byte[] data = "abcde".getBytes();

        fos.write(data);

        //关闭流资源。

        fos.close();

    }

}

给文件中续写和换行

我们直接new FileOutputStream(file)这样创建对象,写入数据,会覆盖原有的文件,那么我们想在原有的文件中续写内容怎么办呢?继续查阅FileOutputStream的API。发现在FileOutputStream的构造函数中,可以接受一个boolean类型的值,如果值true,就会在文件末位继续添加。构造方法

给文件中续写数据和换行,代码演示:

public class FileOutputStreamDemo2 {

    public static void main(String[] args) throws Exception {

        Filefile =new File("c:\\file.txt");

        FileOutputStream    fos =new FileOutputStream(file,true);

        Stringstr ="\r\n" +"";

        fos.write(str.getBytes());

        fos.close();

    }

}

1.2字节输入流InputStream

通过前面的学习,我们可以把内存中的数据写出到文件中,那如何想把硬盘中的数据读到内存中,我们通过InputStream可以实现。InputStream此抽象类,是表示字节输入流的所有类的超类。,定义了字节输入流的基本共性功能方法

int read():读取一个字节并返回,没有字节返回-1.

int read(byte[]): 读取一定量的字节数,并存储到字节数组中,返回读取到的字节数。


InputStream有很多子类,其中子类FileInputStream可用来读取文件内容。

FileInputStream 从文件系统中的某个文件中获得输入字节。

FileInputStream类读取数据read方法:在读取文件中的数据时,调用read方法,实现从文件中读取数据

public class FileInputStreamDemo {

    public static void main(String[] args) throws IOException {

        Filefile =new File("c:\\file.txt");

        //创建一个字节输入流对象,必须明确数据源,其实就是创建字节读取流和数据源相关联。

        FileInputStreamfis =new FileInputStream(file);

        //读取数据。使用 read();一次读一个字节。

        int ch = 0;

        while((ch=fis.read())!=-1){

            System.out.println("ch="+(char)ch);

        // 关闭资源。

        fis.close();

    }

}

读取数据read(byte[])方法

在读取文件中的数据时,调用read方法,每次只能读取一个,太麻烦了,于是我们可以定义数组作为临时的存储容器,这时可以调用重载的read方法,一次可以读取多个字符

public class FileInputStreamDemo2 {

    public static void main(String[] args) throws IOException {

        Filefile =new File("c:\\file.txt");    //演示第二个读取方法,read(byte[]);

       // 创建一个字节输入流对象,必须明确数据源,其实就是创建字节读取流和数据源相关联。

        FileInputStreamfis =newFileInputStream(file);      

        //创建一个字节数组。

        byte[] buf = new byte[1024];//长度可以定义成1024的整数倍。      

        int len = 0;

        while((len=fis.read(buf))!=-1){

            System.out.println(new String(buf,0,len));

        }

        fis.close();

    }

复制文件

原理;读取一个已有的数据,并将这些读到的数据写入到另一个文件中

public class CopyFileTest {

    public static void main(String[] args) throws IOException {

        //1,明确源和目的。

        FilesrcFile =new File("c:\\YesDir\test.JPG");

        FiledestFile =new File("copyTest.JPG");

        //2,明确字节流 输入流和源相关联,输出流和目的关联。

        FileInputStreamfis =new FileInputStream(srcFile);

        FileOutputStreamfos =new FileOutputStream(destFile);

        //3, 使用输入流的读取方法读取字节,并将字节写入到目的中。

        int ch = 0;

        while((ch=fis.read())!=-1){

            fos.write(ch);

        }

        //4,关闭资源。

        fos.close();

        fis.close();

    }

}

上述代码输入流和输出流之间是通过ch这个变量进行数据交换的。上述复制文件有个问题,每次都从源文件读取一个,然后在写到指定文件,接着再读取一个字符,然后再写一个,一直这样下去。效率极低

缓冲数组方式复制文件

上述代码复制文件效率太低了,并且频繁的从文件读数据,和写数据,能不能一次多把文件中多个数据都读进内容中,然后在一次写出去,这样的速度一定会比前面代码速度快。

public class CopyFileByBufferTest {

    public static void main(String[] args) throws IOException {

        FilesrcFile =new File("c:\\YesDir\test.JPG");

        FiledestFile =new File("copyTest.JPG");

        // 明确字节流 输入流和源相关联,输出流和目的关联。

        FileInputStreamfis =new FileInputStream(srcFile);

        FileOutputStreamfos =new FileOutputStream(destFile);

        //定义一个缓冲区。

        byte[] buf = new byte[1024];

        int len = 0;

        while ((len = fis.read(buf))!= -1) {

            fos.write(buf, 0, len);// 将数组中的指定长度的数据写入到输出流中。

        }

        // 关闭资源。

        fos.close();

        fis.close();

    }

}

二,字符流

经过前面的学习,我们基本掌握的文件的读写操作,在操作过程中字节流可以操作所有数据,可是当我们操作的文件中有中文字符,并且需要对中文字符做出处理时怎么办呢?

字节流读取字符的问题

通过以下程序读取带有中文件的文件。 

public class CharStreamDemo {

    public static void main(String[] args) throws IOException {

        //给文件中写中文

        writeCNText();

        //读取文件中的中文

        readCNText();

    }  

    //读取中文

    public static void readCNText() throws IOException {

        FileInputStreamfis=new FileInputStream("c:\\cn.txt");

        int ch = 0;

        while((ch =fis.read())!=-1){

            System.out.println(ch);

        }

    }

    //写中文

    public static void writeCNText() throws IOException {

        FileOutputStreamfos =new FileOutputStream("c:\\cn.txt");

        fos.write("a欢迎你".getBytes());

        fos.close();

    }

}

上面程序在读取含有中文的文件时,我们并没有看到具体的中文,而是看到一些数字,这是什么原因呢?既然看不到中文,那么我们如何对其中的中文做处理呢?要解决这个问题,我们必须研究下字符的编码过程。

字符编码表

我们知道计算机底层数据存储的都是二进制数据,而我们生活中的各种各样的数据,如何才能和计算机中存储的二进制数据对应起来呢?

这时老美他们就把每一个字符和一个整数对应起来,就形成了一张编码表,老美他们的编码表就是ASCII表。其中就是各种英文字符对应的编码。

编码表:其实就是生活中字符和计算机二进制的对应关系表。

1、ascii:一个字节中的7位就可以表示。对应的字节都是正数。0-xxxxxxx

2、iso-8859-1:拉丁码表 latin,用了一个字节用的8位。1-xxxxxxx  负数。

3、GB2312:简体中文码表。包含6000-7000中文和符号。用两个字节表示。两个字节第一个字节是负数,第二个字节可能是正数

GBK:目前最常用的中文码表,2万的中文和符号。用两个字节表示,其中的一部分文字,第一个字节开头是1,第二字节开头是0

GB18030:最新的中文码表,目前还没有正式使用。

unicode:国际标准码表:无论是什么文字,都用两个字节存储。

Java中的char类型用的就是这个码表。char c = 'a';占两个字节。

Java中的字符串是按照系统默认码表来解析的。简体中文版 字符串默认的码表是GBK。

5、UTF-8:基于unicode,一个字节就可以存储数据,不要用两个字节存储,而且这个码表更加的标准化,在每一个字节头加入了编码信息(后期到api中查找)。

能识别中文的码表:GBK、UTF-8;正因为识别中文码表不唯一,涉及到了编码解码问题。

对于我们开发而言;常见的编码GBK  UTF-8  ISO-8859-1

文字--->(数字) :编码。“abc”.getBytes()  byte[]

(数字)--->文字  :解码。byte[] b={97,98,99}  newString(b)

1.1字符输入流Reader

上述程序中我们读取拥有中文的文件时,使用的字节流在读取,那么我们读取到的都是一个一个字节。只要把这些字节去查阅对应的编码表,就能够得到与之对应的字符。API中是否给我们已经提供了读取相应字符的功能流对象,Reader,读取字符流的抽象超类

read():读取单个字符并返回

read(char[]):将数据读取到数组中,并返回读取的个数。

FileReader类

查阅FileInputStream的API,发现FileInputStream 用于读取诸如图像数据之类的原始字节流。要读取字符流,请考虑使用 FileReader。打开FileReader的API介绍。用来读取字符文件的便捷类。此类的构造方法假定默认字符编码和默认字节缓冲区大小都是适当的

FileReader读取包含中文的文件

使用FileReader读取包含中文的文件

public class CharStreamDemo {

    public static void main(String[] args) throws IOException {

        //给文件中写中文

        writeCNText();

        //读取文件中的中文

        readCNText();

    }  

    //读取中文

    public static void readCNText() throws IOException {

        FileReaderfr =new FileReader("D:\\test\\cn.txt");

        int ch = 0;

        while((ch = fr.read())!=-1){

            //输出的字符对应的编码值

            System.out.println(ch);

            //输出字符本身

            System.out.println((char)ch);

        }

    }

    //写中文

    public static void writeCNText() throws IOException {

        FileOutputStreamfos =new FileOutputStream("D:\\test\\cn.txt");

        fos.write("a尚观欢迎你".getBytes());

        fos.close();

    }

1.2字符输出流Writer

既然有专门用于读取字符的流对象,那么肯定也有写的字符流对象,查阅API,发现有一个Writer类,Writer是写入字符流的抽象类。其中描述了相应的写的动作

FileWriter类

查阅FileOutputStream的API,发现FileOutputStream 用于写入诸如图像数据之类的原始字节的流。要写入字符流,请考虑使用 FileWriter。打开FileWriter的API介绍。用来写入字符文件的便捷类。此类的构造方法假定默认字符编码和默认字节缓冲区大小都是可接受的。

FileWriter写入中文到文件中

写入字符到文件中,先进行流的刷新,再进行流的关闭。

public class FileWriterDemo {

    public static void main(String[] args) throws IOException {

        //演示FileWriter 用于操作文件的便捷类。

        FileWriterfw =new FileWriter("d:\\text\\fw.txt");

        fw.write("你好谢谢再见");//这些文字都要先编码。都写入到了流的缓冲区中。

        fw.flush();

        fw.close();

    }

}

复制文本文件

练习:复制文本文件。

思路:

1,既然是文本涉及编码表。需要用字符流。

2,操作的是文件。涉及硬盘。

3,有指定码表吗?没有,默认就行。

操作的是文件,使用的默认码表。使用哪个字符流对象。直接使用字符流操作文件的便捷类。FileReader  FileWriter


public class CopyTextFileTest {

    public static void main(String[] args) throws IOException {

        copyTextFile();

    }

    public static void copyTextFile() throws IOException {

        //1,明确源和目的。

        FileReaderfr =new FileReader("c:\\cn.txt");

        FileWriterfw =new FileWriter("c:\\copy.txt");

        //2,为了提高效率。自定义缓冲区数组。字符数组。

        char[] buf = new char[1024];

        int len = 0;

        while((len=fr.read(buf))!=-1){

            fw.write(buf,0,len);

        }

        /*2,循环读写操作。效率低。

        intch=0;

        while((ch=fr.read())!=-1){

            fw.write(ch);

        }

        */

        //3,关闭资源。

        fw.close();

        fr.close();

    }

}

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,590评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,808评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,151评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,779评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,773评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,656评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,022评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,678评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,038评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,756评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,411评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,005评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,973评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,053评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,495评论 2 343

推荐阅读更多精彩内容