java基础(12)- java的输入/输出(IO)包(2)

来源:互联网 发布:ios 淘宝详情页面 编辑:程序博客网 时间:2024/06/08 19:04

注 : 参考自https://mp.weixin.qq.com/s/MrrN7BDcKMu8SCKs6zij_g

12.1带缓冲的流

12.2datainputstream和dataoutputstream

12.3randomaccessfile


12.1带缓冲的流

一、什么是带缓冲的流

在字符流中可以使用带缓冲的字符流提高文件读写的效率,这种效率的提高是显著的而且有时是必须的。带缓冲提供一个较大的缓冲区,通过减少文件读取或写入次数以提高IO流的效率,比如相比较使用一个int的长度是4byte临时变量,使用而默认的BufferedReader缓冲区大小是4k byte,减少读取的次数是4×1024/8=1024倍。

缓存流可以修饰原始的字节流和字符流,带缓存的字节流类是:java.io.BufferedInputStream和java.io.BufferedOutputStream,而带缓存的字符流是:java.io.BufferedReader和java.io.BufferedWriter。在学习javaIO时会发现该API下存在大量的修饰器模式,修饰器模式下以一种功能”低级”的流去修饰功能更为“高级”流,从而得到更为高级的功能。BufferedReader的构造器有两个构造函数:

BufferedReader(Reader in)

BufferedReader(Reader in, int size)

前面一种方式使用一个Reader的实例去修饰BufferedReader,使用默认大小的缓冲器的带缓冲字符流。而第二种构造缓冲器长度是由size设定。

跟BufferedReader一样,BufferedWriter的构造器也有两种重载的格式:

BufferedWriter(Writer out)

BufferedWriter(Writer out, int size)

字节缓冲输入/输出是一个非常普通的性能优化。Java 的BufferedInputStream 类允许把任何InputStream 类“包装”成缓冲流并使它的性能提高。BufferedInputStream 有两个构造函数:

BufferedInputStream(InputStream inputStream)

BufferedInputStream(InputStream inputStream, int bufSize)

而对应的缓冲输出流BufferedOutputStream

BufferedOutputStream与任何一个OutputStream相同,因为BufferedOutputStream是通过减小系统写数据的时间而提高性能的,可以调用flush( )方法生成缓冲器中待写的数据。下面是两个可用的构造函数:

BufferedOutputStream(OutputStream outputStream)

BufferedOutputStream(OutputStream outputStream, int bufSize)

第一种形式创建了一个使用512字节缓冲器的缓冲流。第二种形式,缓冲器的大小由bufSize参数传入。

在带缓冲的输出流中,一般都增加了flush()方法,flush()方法的作用是确保数据缓冲器确实被写到实际的输出流。用BufferedWriter可以通过减小数据被实际的写到输出流的次数而提高程序的性能。对于字节流缓冲流(buffered stream),通过把内存缓冲器连到输入/输出流扩展一个过滤流类。该缓冲器允许Java对多个字节同时进行输入/输出操作,提高了程序性能。

二、带缓存的字

使用带缓存的字符流修饰原始的字符流,代码如下:

public void input() {try {// 带缓冲的字符流BufferedReader bufreader = new BufferedReader(            new FileReader("e://abc.txt"));while (bufreader.ready()) {// 可以一次读取一行System.out.println(bufreader.readLine());}} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();}}public void output() {try {BufferedWriter writer = new BufferedWriter(           new FileWriter(new File("d://abc.txt")));writer.write("大家好啊!I say:\" \";\r\n");writer.write("不好");writer.flush();//要调用flush方法writer.close();} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();}}

三、带缓存的字

我们已经学过带缓冲的字符流BufferedReader和BufferedWriter,带缓冲的字节流跟前者非常相似,要改写上面的程序非常简单,只需要分别创建BufferedInputStream /BufferedOutputStream对象,将原来代码中的fis和fout进行修饰即可,对大量的文本读取性能有很大提高。示例代码如下:

public void input() {    try {        InputStream input = new FileInputStream("e://abc.txt");        // 如果修饰得是字节流,带缓冲的字节流        BufferedInputStream buinput = new BufferedInputStream(input);        // 每次读取数据的临时变量,每次读32位4byte        int tmp = 0;        // 循环读取,最后读到文件末尾返回-1        while ((tmp = buinput.read()) != -1) {        // 每读一次写一次        System.out.println((char)tmp);        }        buinput.close();        } catch (FileNotFoundException e) {        e.printStackTrace();    } catch (IOException e) {        e.printStackTrace();    }}public void output() {    try {        FileOutputStream output = new FileOutputStream("e://abc.txt");        BufferedOutputStream buoutput = new BufferedOutputStream(output);        buoutput.write('a');        buoutput.write('b');        buoutput.flush();        // 释放资源        buoutput.close();    } catch (FileNotFoundException e) {        e.printStackTrace();    } catch (IOException e) {        e.printStackTrace();    }}


12.2datainputstream和dataoutputstream

datainputstream和dataoutputstream

一、datainputstream和dataoutputstream

使用BufferedInputStream/BufferedOutStream读写的并不是纯二进制数据,因此在有些时候,比如网络互传时会有出现文件无法恢复的问题,除非是纯文本文件。要解决文件流的通用性就必须使用纯二进制的方式读写数据,这时我们可以选择使用DataInputStream

/DataOutputStream对数据使用二进制的方式进行读写,比如以DataInputStream为例,字节入流的API结构]显示了其继承于java.io. FilterInputStream。DataInputStream可以被认为是“更高级”的类,在编程中需要使用修饰其他的InputStream来进行使用,考虑到效率,推荐这个被修饰的类为带缓冲的流,比如:BufferedInputStream。DataOutputStream的情况也是如此,就不需我们再次累述。

DataInputStream的构造器:

DataInputStream(InputStream in)

DataOutputStream的构造器:

DataOutputStream(OutputStream out)

java.io.DataInputStream dis = new java.io.DataInputStream(

new BufferedInputStream(new URL(srcUri).openStream()));

/*生成二进制输出流 ,使用FileOutputStream--修饰-->BufferedOutputStream

--修饰-->DataOutputStream*/

java.io.DataOutputStream dos = new java.io.DataOutputStream(

new BufferedOutputStream(new FileOutputStream(targetFile)));


二、示例代码

    public static void copy(String srcpath, String targetpath) {

DataInputStream input = null;
DataOutputStream output = null;

try {
input = new DataInputStream(new BufferedInputStream(
new FileInputStream(new File(srcpath))));
output = new DataOutputStream(new BufferedOutputStream(
new FileOutputStream(new File(targetpath))));

int temp;

while ((temp = input.read()) != -1) {
output.write(temp);
}

} catch (FileNotFoundException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} finally {
// 释放资源
try {
input.close();
output.flush();
output.close();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}

}


12.3randomaccessfile

randomaccessfile

一、RandomAccessFile讲解

RandomAccessFile类不属于InputStream或者OutputStream分层结构的一部分,它实现了DataInput和DataOutput。RandomAccessFile用于包含了已知长度记录的文件,我们可以使用seek()方法从一条记录移至另一条,可在一个文件里向前或向后移动,然后读取或修改那部分的记录,不像之前的InputStream和OutputStream的实现类那样,每次都从头开始读取。

构造器要求使用者传入一个变量指出对象只是随机读(“r”),还是读写兼施(“rw”)。getFilePointer()用于了解当前在文件的什么地方,seek()用于移至文件内的一个新地点。length()用于判断文件的最大长度。

二、示例代码

public static void main(String[] args) {//mode模式   r只读 rw 读写try {RandomAccessFile rafile=new RandomAccessFile(new File("e://ram.txt"), "rw");    System.out.println("文件长度:"+rafile.length());        //改变光标的位置:    rafile.seek(3L);    System.out.println("光标的位置:"+rafile.getFilePointer());        //主意光标。覆盖字符    rafile.writeBytes("abc");        int temp;    temp=rafile.read();//从当前光标的位置开始往后读取    while(temp!=-1){    System.out.println("光标的位置:"+rafile.getFilePointer());    System.out.println((char)temp);    temp=rafile.read();//读取下一个    }        rafile.close();} catch (FileNotFoundException e) {// TODO Auto-generated catch blocke.printStackTrace();} catch (IOException e) {// TODO Auto-generated catch blocke.printStackTrace();}}




原创粉丝点击