Java IO

Java流(Stream)、文件(File)和IO

Java 流(Stream)、文件(File)和IO

Java.io包几乎包含了所有操作输入、输出需要的类。所有这些流类代表了输入源和输出目标。

Java.io包中的流支持很多种格式,比如:基本类型、对象、本地化字符集等等。

一个流可以理解为一个数据的序列。输入流表示从一个源读取数据,输出流表示向一个目标写数据。

Java为I/O提供了强大的而灵活的支持,使其更广泛地应用到文件传输网络编程中。

但本节讲述最基本的和流与I/O相关的功能。我们将通过一个个例子来学习这些功能。


读取控制台输入

Java的控制台输入由System.in完成。

为了获得一个绑定到控制台的字符流,你可以把System.in包装在一个BufferedReader 对象中来创建一个字符流。

下面是创建BufferedReader的基本语法:

1

2

BufferedReader br = new BufferedReader(new 

                      InputStreamReader(System.in));

BufferedReader对象创建后,我们便可以使用read()方法从控制台读取一个字符,或者用readLine()方法读取一个字符串。


从控制台读取多字符输入

从BufferedReader对象读取一个字符要使用read()方法,它的语法如下:

1

int read( ) throws IOException

每次调用read()方法,它从输入流读取一个字符并把该字符作为整数值返回。 当流结束的时候返回-1。该方法抛出IOException。

下面的程序示范了用read()方法从控制台不断读取字符直到用户输入"q"。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

// 使用 BufferedReader 在控制台读取字符

 

import java.io.*;

 

public class BRRead {

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

   {

      char c;

      // 使用 System.in 创建 BufferedReader 

      BufferedReader br = new BufferedReader(new 

                         InputStreamReader(System.in));

      System.out.println("Enter characters, 'q' to quit.");

      // 读取字符

      do {

         c = (char) br.read();

         System.out.println(c);

      } while(c != 'q');

   }

}

以上实例编译运行结果如下:

1

2

3

4

5

6

7

8

9

Enter characters, 'q' to quit.

123abcq

1

2

3

a

b

c

q


从控制台读取字符串

从标准输入读取一个字符串需要使用BufferedReader的readLine()方法。

它的一般格式是:

1

String readLine( ) throws IOException

下面的程序读取和显示字符行直到你输入了单词"end"。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

// 使用 BufferedReader 在控制台读取字符

import java.io.*;

public class BRReadLines {

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

   {

      // 使用 System.in 创建 BufferedReader 

      BufferedReader br = new BufferedReader(new

                              InputStreamReader(System.in));

      String str;

      System.out.println("Enter lines of text.");

      System.out.println("Enter 'end' to quit.");

      do {

         str = br.readLine();

         System.out.println(str);

      } while(!str.equals("end"));

   }

}

以上实例编译运行结果如下:

1

2

3

4

5

6

7

8

Enter lines of text.

Enter 'end' to quit.

This is line one

This is line one

This is line two

This is line two

end

end

 

控制台输出

在此前已经介绍过,控制台的输出由 print( ) println( )完成。这些方法都由类PrintStream 定义,System.out是该类对象的一个引用。

PrintStream 继承了OutputStream类,并且实现了方法write()。这样,write()也可以用来往控制台写操作。

PrintStream定义write()的最简单格式如下所示:

1

void write(int byteval)

该方法将byteval的低八位字节写到流中。

实例

下面的例子用write()把字符"A"和紧跟着的换行符输出到屏幕:

1

2

3

4

5

6

7

8

9

10

11

import java.io.*;

 

// 演示 System.out.write().

public class WriteDemo {

   public static void main(String args[]) {

      int b; 

      b = 'A';

      System.out.write(b);

      System.out.write('\n');

   }

}

运行以上实例在输出窗口输出"A"字符

1

A

注意:write()方法不经常使用,因为print()和println()方法用起来更为方便。


读写文件

如前所述,一个流被定义为一个数据序列。输入流用于从源读取数据,输出流用于向目标写数据。

下图是一个描述输入流和输出流的类层次图。

下面将要讨论的两个重要的流是FileInputStream FileOutputStream


FileInputStream

该流用于从文件读取数据,它的对象可以用关键字new来创建。

有多种构造方法可用来创建对象。

可以使用字符串类型的文件名来创建一个输入流对象来读取文件:

1

InputStream f = new FileInputStream("C:/java/hello");

也可以使用一个文件对象来创建一个输入流对象来读取文件。我们首先得使用File()方法来创建一个文件对象:

1

2

File f = new File("C:/java/hello");

InputStream f = new FileInputStream(f);

创建了InputStream对象,就可以使用下面的方法来读取流或者进行其他的流操作。

序号

方法及描述

1

public void close() throws IOException{}
关闭此文件输入流并释放与此流有关的所有系统资源。抛出IOException异常。

2

protected void finalize()throws IOException {}
这个方法清除与该文件的连接。确保在不再引用文件输入流时调用其 close 方法。抛出IOException异常。

3

public int read(int r)throws IOException{}
这个方法从InputStream对象读取指定字节的数据。返回为整数值。返回下一字节数据,如果已经到结尾则返回-1

4

public int read(byte[] r) throws IOException{}
这个方法从输入流读取r.length长度的字节。返回读取的字节数。如果是文件结尾则返回-1。

5

public int available() throws IOException{}
返回下一次对此输入流调用的方法可以不受阻塞地从此输入流读取的字节数。返回一个整数值。

除了InputStream外,还有一些其他的输入流,更多的细节参考下面链接:

·        ByteArrayInputStream

·        DataInputStream


FileOutputStream

该类用来创建一个文件并向文件中写数据。

如果该流在打开文件进行输出前,目标文件不存在,那么该流会创建该文件。

有两个构造方法可以用来创建FileOutputStream 对象。

使用字符串类型的文件名来创建一个输出流对象:

1

OutputStream f = new FileOutputStream("C:/java/hello")

也可以使用一个文件对象来创建一个输出流来写文件。我们首先得使用File()方法来创建一个文件对象:

1

2

File f = new File("C:/java/hello");

OutputStream f = new FileOutputStream(f);

创建OutputStream 对象完成后,就可以使用下面的方法来写入流或者进行其他的流操作。

序号

方法及描述

1

public void close() throws IOException{}
关闭此文件输入流并释放与此流有关的所有系统资源。抛出IOException异常。

2

protected void finalize()throws IOException {}
这个方法清除与该文件的连接。确保在不再引用文件输入流时调用其 close 方法。抛出IOException异常。

3

public void write(int w)throws IOException{}
这个方法把指定的字节写到输出流中。

4

public void write(byte[] w)
把指定数组中w.length长度的字节写到OutputStream中。

除了OutputStream外,还有一些其他的输出流,更多的细节参考下面链接:

·        ByteArrayOutputStream

·        DataOutputStream

实例

下面是一个演示InputStream和OutputStream用法的例子:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

import java.io.*;

 

public class fileStreamTest{

 

   public static void main(String args[]){

    

   try{

      byte bWrite [] = {11,21,3,40,5};

      OutputStream os = new FileOutputStream("test.txt");

      for(int x=0; x < bWrite.length ; x++){

         os.write( bWrite[x] ); // writes the bytes

      }

      os.close();

      

      InputStream is = new FileInputStream("test.txt");

      int size = is.available();

 

      for(int i=0; i< size; i++){

         System.out.print((char)is.read() + "  ");

      }

      is.close();

   }catch(IOException e){

      System.out.print("Exception");

   }

   }

}

上面的程序首先创建文件test.txt,并把给定的数字以二进制形式写进该文件,同时输出到控制台上。

以上代码由于是二进制写入,可能存在乱码,你可以使用以下代码实例来解决乱码问题:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

//文件名 :fileStreamTest2.java

import java.io.*;

 

public class fileStreamTest2{

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

         

        File f = new File("a.txt");

        FileOutputStream fop = new FileOutputStream(f);

        // 构建FileOutputStream对象,文件不存在会自动新建

         

        OutputStreamWriter writer = new OutputStreamWriter(fop, "UTF-8");

        // 构建OutputStreamWriter对象,参数可以指定编码,默认为操作系统默认编码,windows上是gbk

         

        writer.append("中文输入");

        // 写入到缓冲区

         

        writer.append("\r\n");

        //换行

         

        writer.append("English");

        // 刷新缓存冲,写入到文件,如果下面已经没有写入的内容了,直接close也会写入

         

        writer.close();

        //关闭写入流,同时会把缓冲区内容写入文件,所以上面的注释掉

         

        fop.close();

        // 关闭输出流,释放系统资源

 

        FileInputStream fip = new FileInputStream(f);

        // 构建FileInputStream对象

         

        InputStreamReader reader = new InputStreamReader(fip, "UTF-8");

        // 构建InputStreamReader对象,编码与写入相同

 

        StringBuffer sb = new StringBuffer();

        while (reader.ready()) {

            sb.append((char) reader.read());

            // 转成char加到StringBuffer对象中

        }

        System.out.println(sb.toString());

        reader.close();

        // 关闭读取流

         

        fip.close();

        // 关闭输入流,释放系统资源

 

    }

}


文件和I/O

还有一些关于文件和I/O的类,我们也需要知道:

·        File Class(类)

·        FileReader Class(类)

·        FileWriter Class(类)


Java中的目录

创建目录:

File类中有两个方法可以用来创建文件夹:

·        mkdir( )方法创建一个文件夹,成功则返回true,失败则返回false。失败表明File对象指定的路径已经存在,或者由于整个路径还不存在,该文件夹不能被创建。

·        mkdirs()方法创建一个文件夹和它的所有父文件夹。

下面的例子创建 "/tmp/user/java/bin"文件夹:

1

2

3

4

5

6

7

8

9

10

import java.io.File;

 

public class CreateDir {

   public static void main(String args[]) {

      String dirname = "/tmp/user/java/bin";

      File d = new File(dirname);

      // 现在创建目录

      d.mkdirs();

  }

}

编译并执行上面代码来创建目录"/tmp/user/java/bin"。

注意:Java在UNIX和Windows自动按约定分辨文件路径分隔符。如果你在Windows版本的Java中使用分隔符(/) ,路径依然能够被正确解析。


读取目录

一个目录其实就是一个File对象,它包含其他文件和文件夹。

如果创建一个File对象并且它是一个目录,那么调用isDirectory( )方法会返回true。

可以通过调用该对象上的list()方法,来提取它包含的文件和文件夹的列表。

下面展示的例子说明如何使用list()方法来检查一个文件夹中包含的内容:

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

import java.io.File;

 

public class DirList {

   public static void main(String args[]) {

      String dirname = "/tmp";

      File f1 = new File(dirname);

      if (f1.isDirectory()) {

         System.out.println( "Directory of " + dirname);

         String s[] = f1.list();

         for (int i=0; i < s.length; i++) {

            File f = new File(dirname + "/" + s[i]);

            if (f.isDirectory()) {

               System.out.println(s[i] + " is a directory");

            } else {

               System.out.println(s[i] + " is a file");

            }

         }

      } else {

         System.out.println(dirname + " is not a directory");

    }

  }

}

以上实例编译运行结果如下:

1

2

3

4

5

6

7

8

Directory of /tmp

bin is a directory

lib is a directory

demo is a directory

test.txt is a file

README is a file

index.html is a file

include is a directory

读取控制台输入

从控制台读取多字符输入

从控制台读取字符串

控制台输出

4.1 实例

读写文件

FileInputStream

FileOutputStream

7.1 实例

文件和I/O

Java中的目录

9.1 创建目录:

10 读取目录

 

packagecom.imooc.io;

 

importjava.io.File;

importjava.io.IOException;

 

publicclass FileDemo {

 

    /**

     *@param args

     */

    public static void main(String[] args) {

       // 了解构造函数的情况  查帮助

        File file = newFile("E:\\javaio\\imooc");

        //System.out.println(file.exists());

        if(!file.exists())

            file.mkdir();//file.mkdirs()

        else

            file.delete();

        //是否是一个目录 如果是目录返回true,如果不是目录or目录不存在返回的是false

        System.out.println(file.isDirectory());

        //是否是一个文件

        System.out.println(file.isFile());

       

        //File file2 = newFile("e:\\javaio\\日记1.txt");

        File file2 = newFile("e:\\javaio","日记1.txt");

        if(!file2.exists())

            try {

                file2.createNewFile();

            } catch (IOException e) {

                // TODOAuto-generated catch block

                e.printStackTrace();

            }

        else

            file2.delete();

         //常用的File对象的API

        System.out.println(file);//file.toString()的内容

        System.out.println(file.getAbsolutePath());

        System.out.println(file.getName());

        System.out.println(file2.getName());

        System.out.println(file.getParent());

        System.out.println(file2.getParent());

        System.out.println(file.getParentFile().getAbsolutePath());

    }

 

}

java.io.File类用于表示文件(目录)

File类只用于表示文件(目录)的信息(名称、大小等),不能用于文件内容的访问

 

RandomAccessFile java提供的对文件内容的访问,既可以读文件,也可以写文件。

RandomAccessFile支持随机访问文件,可以访问文件的任意位置

 

(1)java文件模型

  在硬盘上的文件是bytebyte byte存储的,是数据的集合

(2)打开文件

  有两种模式"rw"(读写)  "r"(只读)

  RandomAccessFile raf = newRandomeAccessFile(file,"rw")

  文件指针,打开文件时指针在开头 pointer = 0;

(3) 写方法

   raf.write(int)--->只写一个字节(后8位),同时指针指向下一个位置,准备再次写入

(4)读方法

   int b =raf.read()--->读一个字节

(5)文件读写完成以后一定要关闭(Oracle官方说明)

 

 

序列化与基本类型序列化

1)将类型int 转换成4byte或将其他数据类型转换成byte的过程叫序列化

     数据---->nbyte

2)反序列化

    将n个byte 转换成一个数据的过程

    nbyte---> 数据

3)RandomAccessFile提供基本类型的读写方法,可以将基本类型数据

   序列化到文件或者将文件内容反序列化为数据

 IO流(输入流、输出流)

 字节流、字符流

 1.字节流

 1)InputStream、OutputStream

   InputStream抽象了应用程序读取数据的方式

   OutputStream抽象了应用程序写出数据的方式

 2)EOF =End   读到-1就读到结尾

 3)输入流基本方法

   int  b = in.read();读取一个字节无符号填充到int低八位.-1是 EOF

   in.read(byte[] buf)

   in.read(byte[] buf,int start,intsize)

4)输出流基本方法

  out.write(int b)  写出一个byte到流,b的低8位

  out.write(byte[] buf)将buf字节数组都写入到流

  out.write(byte[] buf,int start,intsize)

 

 5)FileInputStream--->具体实现了在文件上读取数据

 6)FileOutputStream 实现了向文件中写出byte数据的方法

 7)DataOutputStream/DataInputStream

    对"流"功能的扩展,可以更加方面的读取int,long,字符等类型数据

   DataOutputStream

       writeInt()/writeDouble()/writeUTF()

 

 8)BufferedInputStream&BufferedOutputStream

 这两个流类位IO提供了带缓冲区的操作,一般打开文件进行写入

 或读取操作时,都会加上缓冲,这种流模式提高了IO的性能

 从应用程序中把输入放入文件,相当于将一缸水倒入到另一个缸中:

 FileOutputStream--->write()方法相当于一滴一滴地把水“转移”过去

 DataOutputStream-->writeXxx()方法会方便一些,相当于一瓢一瓢把水“转移”过去

 BufferedOutputStream--->write方法更方便,相当于一飘一瓢先放入桶中,再从桶中倒入到另一个缸中,性能提高了

  

  

 2.字符流

 1) 编码问题

 2)认识文本和文本文件

 java的文本(char)是16位无符号整数,是字符的unicode编码(双字节编码)

 文件是bytebyte byte ...的数据序列

文本文件是文本(char)序列按照某种编码方案(utf-8,utf-16be,gbk)序列化为byte的存储结果

3)字符流(ReaderWriter)---->操作的是文本文本文件

字符的处理,一次处理一个字符

字符的底层任然是基本的字节序列

字符流的基本实现

   InputStreamReader   完成byte流解析为char流,按照编码解析

   OutputStreamWriter  提供char流到byte流,按照编码处理  

  

   FileReader/FileWriter

 字符流的过滤器

   BufferedReader   ---->readLine 一次读一行

   BufferedWriter/PrintWriter   ---->写一行   

  

  

3.对象的序列化,反序列化

1)对象序列化,就是将Object转换成byte序列,反之叫对象的反序列化

2)序列化流(ObjectOutputStream),是过滤流----writeObject

   反序列化流(ObjectInputStream)---readObject

 

3)序列化接口(Serializable)

   对象必须实现序列化接口,才能进行序列化,否则将出现异常

   这个接口,没有任何方法,只是一个标准

 

4) transient关键字

    private voidwriteObject(java.io.ObjectOutputStream s)

               throws java.io.IOException

    private void readObject(java.io.ObjectInputStream s)

               throws java.io.IOException,ClassNotFoundException

              

   分析ArrayList源码中序列化和反序列化的问题

 

5)序列化中 子类和父类构造函数的调用问题

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值