day_25I/O输入输出

 

目的:

 

在变量、数组和对象中存储数据是暂时的,程序结束后它们就会丢失。

 

为了能够永久地保存程序创建的数据,需要将其保存在磁盘文件中,这样以后就可以在其他程序中使用它们。

Java的I/O技术可以将数据保存到文本文件、二进制文件甚至是ZIP压缩文件中,以达到永久性保存数据的要求。

掌握I/O处理技术能够提高对数据的处理能力。

 

了解Java中的流概念

了解Java中输入/输出流的分类

掌握文件输入/输出的使用方法

掌握ZIP压缩输入/输出流的应用

 

12.1流概述

 

  流是一组有序的数据序列,根据操作的类型,可分为输入流和输出流两种。

 

I/O提供了一条通道程序,可以使用这条通道把源中的字节序列送到目的地。

 

虽然I/O流经常与磁盘文件存取有关,但是程序的源和目的也可以是键盘、鼠标、内存或显示器窗口等。

 

12.2 输入/输出流

 

java.io包

输入流类:

       抽象类InputStream(字节输入流)或抽象类Reader(字符输入流)的子类;

输出流类:

       抽象类OutputStream(字节输出流)或抽象类Writer(字符输出流)的子类。

 

12.2.1输入流

java.lang.Object

java.io.InputStream

 

 

Java中的字符是Unicode编码,是双字节的。

InputStream是用来处理字节的,在处理字符文本时不是很方便。

Java为字符文本的输入提供了专门一套单独的类Reader,但是Reader类

并不是InputStream类的替换者,只是在处理字符串时简化了编程。

 

 

Reader类是字符输入流的抽象类,所有字符输入流的实现都是它的子类。

 

 

12.2.2输出流

OutputStream类是字节输入流的抽象类,此抽象类表示输出字节流的所有类的超类。

OutputStream类中的所有方法均返回void,在遇到错误时会引发IOException异常。

 

Writer 类是字符输出流的抽象类,所有字符输出类的实现都是它的子类。

 

 

12.2.3 案例

显示指定类型的文件

 

 

 

12.3 File类

 

File类是io包中唯一代表磁盘文件本身的对象。

可以通过调用File类中的方法,实现创建、删除、重命名文件等。

 

File类的对象主要用来获取文件本身的一些信息,如文件所在的目录、文件的长度、文件读写权限等。

 

数据流可以将数据写入到文件中,而文件也是数据流最常用的数据媒体。

 

12.3.1文件的创建于删除

 

构造方法

1.File(String pathname) 

          通过将给定路径名字符串转换为抽象路径名来创建一个新 File 实例。

 

语法格式:

new File(String pathname)

 

pathname: 是指路径名称(包含文件名)

 

File file = new File("d:/1.txt");

 

2.File(String parent, String child) 

          根据 parent 路径名字符串和 child 路径名字符串创建一个新 File 实例。

 

该构造方法根据定义的父路径和子路径字符串(包含文件名)创建一个新的File对象。

new File(String parent,String child)  

parent:父路径字符串 

child:   子路径字符串

 

 

3.File(File parent, String child) 

          根据 parent 抽象路径名和 child 路径名字符串创建一个新 File 实例。

 

该构造方法根据parent抽象路径名和child路径名字符串创建一个新File实例。

 

new File(File f,String child)

f:父路径对象

child:子路径字符串

 

注意:

       对于windows平台,包含盘符的路径名前缀由驱动器号和一个“:”组成。

如果是绝对路径名,还可能后跟"\\"。

 

12.3.1.1代码案例

package day25;

 

import java.io.File;

import java.io.IOException;

 

public class FileTest {

public static void main(String[] args) {

 

/*File(File?parent, String?child) 

*根据 parent 抽象路径名和 child 路径名字符串创建一个新 File 实例

*在项目中创建FileTest类,在主方法中判断文件夹是否存在..txt文件,

*如果文件存在将其删除,不存在则创建该文件 */

File file = new File("E:/zhuhw_study","test.txt");

if(file.exists()) {

file.delete();

System.out.print("文件已被删除");

}else{

try {

file.createNewFile();

System.out.print("文件已被创建");

} catch (IOException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

 

}

}

}

 

运行结果:

文件已被创建

 

12.3.2获取文件信息

 

 

代码案例:

package day25;

 

import java.io.File;

import java.io.IOException;

 

public class FileTest {

public static void main(String[] args) {

 

/*获取文件夹下的..txt文件的文件名、文件长度、并判断该文件是否是隐藏文件。*/

   File f2 = new File("E:/zhuhw_study","test2.txt");

   System.out.println("文件的文件名:  "+f2.getName());

   System.out.println("文件的长度: "+f2.length());

   System.out.println("文件是否是隐藏文件: "+f2.isHidden());

}

}

运行结果:

文件的文件名:  test2.txt

文件的长度: 0

文件是否是隐藏文件: false

 

12.4文件输出/输出流

 

12.4.1基本概念

如果将数据永久保存,可使用文件输入/输出流与指定的文件建立连接,将需要的数据永久保存到文件中。

java.lang.Object

 java.io.InputStream

    java.io.FileInputStream

 

java.lang.Object

   java.io.OutputStream

     java.io.FileOutputStream

 

都是用来操作磁盘文件的。

 

12.4.1代码案例

 

使用FileOutputStream类向文件work写入信息,然后通过FileInputStream类将work文件中的数据读取到控制台上。

 

 package day25;

import java.io.File;

import java.io.FileInputStream;

import java.io.FileOutputStream;

public class FileTest {

public static void main(String[] args){

/*使用FileOutputStream类向文件word写入信息,

然后通过FileInputStream类将文件中的数据读取到控制台上。*/

File file = new File("G:/asiainfo","test_zhuhw");

try {

FileOutputStream fio = new FileOutputStream(file);

byte b[] = "你好呀,我是瓦询...".getBytes();

fio.write(b);

fio.close();

} catch (Exception e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

try {

FileInputStream fii = new FileInputStream(file);

byte byt[] = new byte[1024];

/*

* 从此输入流中将 byte.length 个字节的数据读入一个 byte 数组中。

* public int read(byte[] b)

* b - 存储读取数据的缓冲区。

* 返回:读入缓冲区的字节总数,如果因为已经到达流末尾

* 而没有更多的数据,则返回 -1 */

int len = fii.read(byt);

/*public String(byte[] bytes,

             int offset,

             int length)

通过使用平台的默认字符集解码指定的 byte 子数组,

构造一个新的 String。新 String 的长度是字符集的函数,

因此可能不等于该子数组的长度。

参数:

bytes - 要解码为字符的 byte

offset - 要解码的第一个 byte 的索引

length - 要解码的 byte 数 

           */

String s = new String(byt,0,len);

System.out.print("文件中的信息是: "+s);

} catch (Exception e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}

}

运行结果:

文件中的信息是: 你好呀,我是瓦询...

 

 

一个被打开的流有可能会用尽系统资源,这取决于平台的和实现。

如果没有将打开的流关闭,当另一个程序试图打开另一个流时,这些资源可能会得不到。

 

 

 

12.4.2 FileReader 和 FileWriter

 

另一个资料看到的,好理解

掌握流的基本概念

掌握Java中各种常用输出输入流的基本用法

能够熟练地编写应用程序解决文件的读写问题

8.1流的基本概念

“流”可以被理解为管道。

管道:两个端口,一端与数据源(当输入数据时)或数据宿(当输出数据时)相连,另一端与程序相连。

在两端口,“管道”在读写数据时能够应付数据源和数据宿的多样性,消化掉因数据源/宿的多样性带来的数据读/写的复杂性;

而在与程序相连的端口,“管道”提供了输入输出的统一操作接口。

对管道的统一而简单的操作,这样就大大降低了输入输出的复杂性。

有了流,程序和外界的数据交换都可通过流实现。

分类:

       根据流中的数据传输的方向,将流分为输入流和输出流。

      根据“管道”里流动的数据类型,将分为字符流(Character Streams)和字节流(Byte Streams)

提示:

       在java的底层操作中,所有的输入输出都是以字节形式进行的。

8.2 字节流

java中的字节流是以字节(byte)为基本处理单位,用于对二进制数据进行读写操作。

对应输入字节流会和输出字节流的两个顶层的抽象类,分别是InputStream和OutputStream。

8.3字符流

java的字符流是以16位的Unicode码表示的字符为基本处理单位。

对应输入字符流和输出字符流的两个顶层的抽象类,分别是Reader和Writer。

8.4 File类

文件是许多程序的基本数据源和数据宿,是保存永久数据和共享信息的媒体。

在Java中,目录页被当作文件,可以用list方法列出目录中的文件名。

8.5 System.in 和System.out对象

可以在程序中直接使用而不用重新定义自己的流对象,因为它们都是静态成员。

8.6 FileInputStream类和FileOutputStream类

通常所使用的文件中,有很多文件是二进制文件,它们以字节作为数据处理的单位。

8.7FileReader类和FileWriter类

因为大多数程序涉及文件读写,,所以FileReader类是一个经常用到的类,可以在一个指定文件上实例化一个文件输入流,

利用流提供的方法从文件中读取一个字符或者一组数组。

最重要的方法是read。

由FileWriter类可以实例化一个文件输出流,并提供向文件中写入一个字符或者一组数据的方法。

8.8过滤流

连接:

https://www.ibm.com/developerworks/cn/java/j-lo-javaio/

星期日, 十一月 29, 2015 15:46:12

5、带缓存的输入输出流

5.1 缓存可以说S是I/O的一种性能优化。

缓存流为I/O流增加了内存缓存区。

5.1 BufferedInputStream与BufferedOutputStream类

BufferedInputStream类可以对任何的InputStream类进行带缓存区的包装以达到性能优化。

文件--->InputStream-->BufferedInputStream--->程序(目的地)

注意:

        BufferedOutputStream有一个flush()方法用来将缓存区的数据强制输出完--->刷新。

flush()方法只对使用缓存区的OutputStream类的子类有效。

当调用close()方法时,系统在关闭流之前,也会将缓存区中信息刷新到磁盘文件中。

5.2 BufferedReader与BufferedWriter类

BufferedReader与BufferedWriter类 分别继承Reader类与Writer类。

这两个类同样具有内部缓存机制,并可以以行为单位进行输出。

字符数据(程序)-->BufferedWriter-->OuterputStreamWriter-->OutputStream-->文件

注意:

        在使用BufferedWriter类的writer()方法时,数据并没有立刻被写入至输出流中,而是首先进入缓存区中。

如果想立刻将缓存区中的数据写入输出流中,一定要调用flush()方法。

5.3案例

向指定的磁盘中写入数据,并通过BufferedReader类将文件中的信息分行显示。

package day29;

import java.io.BufferedReader;

import java.io.BufferedWriter;

import java.io.File;

import java.io.FileNotFoundException;

import java.io.FileReader;

import java.io.FileWriter;

import java.io.IOException;

public class BufferTest {

public static void main(String[] args) {

/*向指定的磁盘中写入数据,

并通过BufferedReader类将文件中的信息分行显示。*/

String content[] = {"好久不见","最近好吗","瓦询..."};//定义字符串数组

File file = new File("G:/word.txt");//创建文件对象

try {

FileWriter fw = new FileWriter(file); //创建FileWriter对象

BufferedWriter bufw = new BufferedWriter(fw);//创建BufferedWriter类对象

for(int k = 0;k < content.length; k++) {     

bufw.write(content[k]);     //将字符串数组中数组中元素写入到磁盘文件中

bufw.newLine(); //将数组中的单个元素以单行的形式写入文件

}

bufw.close();        //将BufferedWriter流关闭

fw.close();          //将FileWriter流关闭

} catch (IOException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

try {

FileReader fr = new FileReader(file);//创建FileReader类对象

BufferedReader bufr = new BufferedReader(fr);//创建BufferedReader类对象

String s = null;                      //创建字符串对象

int i = 0;

while((s = bufr.readLine() )!= null){//如果文件的文本行数不为null,则进入循环

i++;

System.out.println("第"+i+"行  : "+s); //输出文件数据

}

bufr.close();                 //将BufferedReader流关闭

fr.close();                   //将FileReader流关闭

} catch (Exception e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}

}

运行结果:

第1行  : 好久不见

第2行  : 最近好吗

第3行  : 瓦询...

6、数据输入输出流

   6.1 数据输入输出流(DataInputStream类与DataOutputStream类)允许应用程序以与机器无关的方式

从底层输入流中读取基本Java数据类型。= 当读取一个数据时,不必再关心这个数值应当是什么字节。

   6.2 代码案例

 分别通过DataOutputStream类的writeUTF()、writeChars()、writeBytes()方法向指定的磁盘文件中输入数据,

并通过DataInputStream类的readUTF()方法将写入的数据输出到控制台上。

package day29;

import java.io.DataInputStream;

import java.io.DataOutputStream;

import java.io.FileInputStream;

import java.io.FileNotFoundException;

import java.io.FileOutputStream;

import java.io.OutputStream;

public class DataStreamTest {

public static void main(String[] args) {

/*分别通过DataOutputStream类的

* writeUTF()、writeChars()、writeBytes()方法向指定的磁盘文件中输入数据,

* 并通过DataInputStream类的readUTF()方法将写入的数据输出到控制台上。*/

try {

/*FileOutputStream(File file) 

* 创建一个向指定 File 对象表示的文件中写入数据的文件输出流。*/

FileOutputStream fs = new FileOutputStream("G:/word.txt");

/*DataOutputStream(OutputStream out) 

*创建一个新的数据输出流,将数据写入指定基础输出流。*/

DataOutputStream ds = new DataOutputStream(fs);

ds.writeUTF("使用writeUTF()方法写入数据");  //写入磁盘数据

ds.writeChars("使用writeChars()方法写入数据");

ds.writeBytes("使用writeBytes()方法写入数据");

ds.close();

/*DataInputStream(InputStream in) 

* 使用指定的底层 InputStream 创建一个 DataInputStream。*/

FileInputStream fis = new FileInputStream("G:/word.txt");

DataInputStream dis = new DataInputStream(fis);

System.out.print(dis.readUTF());  //将文件数据输出

} catch (Exception e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}

}

运行结果:

使用writeUTF()方法写入数据

7、ZIP压缩输入输出流

可以节省存储空间。java.util.zip包中的ZipOutputStream类与ZipInputSream类来实现文件的压缩/解缩。

如果要将文件内容写到zip文件内,必须先写入对应于该文件的“目录进入点”,

并且把要写入文件内容的位置移到此进入点所指的位置,然后再写入文件内容。

如果要从zip压缩管理文件内读取某个文件,要先找到对应文件的"目录进入点"(从他可知该文件在zip文件内的位置),

才能读取这个文件内容。

7.1压缩文件

利用ZipOutputStream类对象,可将文件压缩为“.zip”文件。

构造函数:

      ZipOutputStream(OutputStream out);

7.2代码案例

本例在zip()方法中实现使用ZipOutputStream类对文件进行压缩,在主方法中调用该方法。

package day29;

import java.io.File;

import java.io.FileInputStream;

import java.io.FileOutputStream;

import java.util.zip.ZipEntry;

import java.util.zip.ZipOutputStream;

public class ZipTest {

/*本例在zip()方法中实现使用ZipOutputStream类对文件进行压缩,在主方法中调用该方法。*/

public static void main(String[] args) {

ZipTest book = new ZipTest();  //创建本例对象

try {

book.zip("G:/hello.zip", new File("G:/hello")); //调用方法,参数为压缩后文件与要压缩文件

System.out.println("压缩完成");//输出信息

} catch (Exception e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}

private void zip(String zipFileName,File inputFile) throws Exception{

FileOutputStream fos = new FileOutputStream(zipFileName);

ZipOutputStream out  = new ZipOutputStream(fos);//创建ZipOutputStream类对象

zip(out,inputFile,"");//调用方法

System.out.println("压缩中...");

out.close();//将流关闭

}

private void zip(ZipOutputStream out,File f,String base) throws Exception{//方法重载

if(f.isDirectory()) { //测试此抽象路径名表示的文件是否是一个目录

File[] f1 = f.listFiles(); //获取路径数组

out.putNextEntry(new ZipEntry(base+"/"));//写入此目录的entry

base = base.length() == 0?"":base+"/";//判断参数是否为空

for(int i =0;i<f1.length;i++){//循环遍历数组中文件

zip(out,f1[i],base+f1[i]);

}

}else{

out.putNextEntry(new ZipEntry(base));//创建新的进入点

FileInputStream in = new FileInputStream(f);//创建FileInput对象

int b;

System.out.println(base);

while((b = in.read())!=-1) {//如果没到达流的尾部

out.write(b);  //将字节写入当前zip条目

}

in.close(); //关闭流

}

}

}

运行结果:

G:\hello\hello1.txt

G:\hello\hello2.txt

压缩中...

压缩完成

7.3 解压缩zip文件

ZipInputStream类可读取zip压缩格式的文件,包括对已压缩和未压缩条目的支持(entry).

7.4代码案例

创建类,通过ZipInputStream类将上例生成的压缩文件解压到指定文件夹中。

package day29;

import java.io.File;

import java.io.FileInputStream;

import java.io.FileNotFoundException;

import java.util.zip.ZipEntry;

import java.util.zip.ZipInputStream;

public class Decompressing {

public static void main(String[] args){

ZipInputStream zin;//创建ZipInputStream对象

try {

zin = new ZipInputStream(new FileInputStream("G:/hello.zip"));

                             //实例化对象,指明要进行解压的文件

ZipEntry entry = zin.getNextEntry();//获取下一个ZipEntry

while(((entry = zin.getNextEntry())!=null)&&!entry.isDirectory()) {

             //如果entry不为空,并不在同一目录下

File file = new File(entry.getName());//获取文件目录

if(!file.exists()) { //如果该文件不存在

file.createNewFile();//创建该文件

}

zin.closeEntry();//关闭当前entry

System.out.println(entry.getName()+"解压成功");

}

zin.close();  //关闭流

} catch (Exception e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

}

}

运行结果:

G:\hello\hello1.txt解压成功

G:\hello\hello2.txt解压成功

本例是通过此ZipEntry的getName()方法,得知此文件名称(含path),借此来决定压缩之后的目录和文件名。

使用ZipInputStream类来解压文件,必须先使用getNextEntry()来获取其内的第一个ZipEntry。

星期日, 十一月 29, 2015 18:45:35

猜你喜欢

转载自yuzhouxiner.iteye.com/blog/2259243