收藏 分享(赏)

JAVA第12章IO编程.ppt

上传人:11xg27ws 文档编号:12236762 上传时间:2021-12-04 格式:PPT 页数:42 大小:286.50KB
下载 相关 举报
JAVA第12章IO编程.ppt_第1页
第1页 / 共42页
JAVA第12章IO编程.ppt_第2页
第2页 / 共42页
JAVA第12章IO编程.ppt_第3页
第3页 / 共42页
JAVA第12章IO编程.ppt_第4页
第4页 / 共42页
JAVA第12章IO编程.ppt_第5页
第5页 / 共42页
点击查看更多>>
资源描述

1、第12章 I/O编程,本章学习重点,流的概念。 InputStream和OutputStream及派生类。 Reader和Writer及其派生字符流类。 File类和RandomAccessFile类的应用。 java.io包的包装技术和设计思想。,I/O,计算机程序的最一般模型其实可以归纳为:输入、计算和输出。 输入和输出是人机交互的重要手段,一个设计合理的程序应该首先允许用户根据具体情况输入不同的数据,然后经过程序算法的计算处理,最后以用户容易接受的方式输出结果。,12.2 流的概念,流(Stream)是对数据传送的一种抽象,当预处理数据从外界“流入”程序中,就称之为输入流,相反,当程序中

2、的结果数据“流到”外界(如显示屏幕、文件等)时,就称之为输出流 输入或输出其实是从程序角度来看待的。 在Java类库中,I/O(输入和输出)部分的内容不少,这点看看JDK的java.io包就知道了,它涉及的主要关键类有:InputStream、OutputStream、Reader、Writer和File等。,Java I/O类,InputStream和OutputStream类是用来处理字节(8位)流的。 Reader和Writer类用来处理字符(16位)流。 File类则用来处理文件。,标准输出,System.out是标准输出流对象,可以通过调用它的println()、print()或wr

3、ite()方法来实现对各种数据的输出显示。,【例 12-1】标准输出方法举例。,标准输入,System.in是标准输入流对象,可以通过调用它的read()方法来实现从键盘读入数据的功能。 由于输入比输出容易出错,Java对输入操作强制设置了异常保护,程序中必须抛出异常或捕获异常,否则程序将不能编译通过。,【例 12-2】标准输入方法举例。,Hint,对于多数程序设计语言(如C和Pascal)来说,它处理的一般字符都是单字节的,而对于Java来说,情况比较特别,当用户输入一般字符(此时为单字节)给Java程序后,若程序中用来存放该字符的数据类型为char时,原本的单字节会自动在高位补扩充为双字节

4、进行存储。 Java采用双字节存储原本为单字节的一般字符,主要是为了将一般字符与其他字符(如汉字字符)统一起来,方便处理。 后面介绍的(Unicode)字符流,即指双字节流。,标准输入功能扩充,原本System.in标准输入流对象只能提供以字节为单位的数据输入,通过引入InputStreamReader和BufferedReader类的对象对其进行两次包装(第一次将System.in对象包装为reader对象的内嵌成员,第二次又将reader对象包装为input对象的成员),这样,就可以使用BufferedReader类提供的readLine()方法,实现以行为单位的字符串输入功能。当获取字符

5、串数据后,还可以根据具体的数据类型进行相应转换。,【例 12-3】扩充的标准输入方法。,【例 12-4】用户输入类MyInput。,【例 12-5】测试MyInput用户输入类。,为了避免不同地方需要进行交互式输入时每次都要重新编写包装语句,建议读者:将上述常用交互式输入单独定义为一个用户输入类MyInput,并将其放置到用户自定义类包myPackage中,以后各个程序或者程序的不同地方就可以方便地进行交互式输入了。,12.3 字节流,以字节为处理单位的流称为字节流,字节流相应地分为字节输入流和字节输出流两种。,InputStream,所有字节输入流的基类为InputStream,它是一个从O

6、bject类直接继承而来的抽象类,类中声明有多个用于字节输入的方法,为其他字节输入流派生类奠定了一个基础,它与其他派生类的继承关系如下图所示 :,ByteArrayInputStream,ByteArrayInputStream输入流类含有四个成员变量:buf 、count、 mark和 pos。buf为字节数组缓冲区,用来存放输入流;count为计数器,记录输入流数据的字节数;mark用来做标记,以实现重读部分输入流数据;pos为位置指示器,指明当前读指针的位置,即已读取count-1个字节的数据。 ByteArrayInputStream输入流类提供的方法基本上与它的基类InputStre

7、am是一样的,因此,ByteArrayInputStream可以说是一个比较简单和基础的字节输入流类。,FileInputStream,FileInputStream类是用来实现从文件中读取字节流数据的,它也是从抽象类InputStream直接继承而来,不过,有些方法,如mark()和reset()等,它并不支持,因为FileInputStream输入流只能实现文件的顺序读取。 另外,FileInputStream既然属于字节输入流类,那么它就不适合来读取字符文件,而适合读取字节文件(如图像文件)。字符文件的读取可以采用后面要介绍的字符输入流类FileReader。,【例 12-6】测试Fil

8、eInputStream文件输入流类。 import java.io.*; public class TestFileInputStream public static void main(String args) throws IOException try /创建文件输入流对象fis FileInputStream fis = new FileInputStream(data.dat); byte buf = new byte128; int count; /记录实际读取字节数 count=fis.read(buf); /从文件输入流fis中读取字节数据 System.out.printl

9、n(共读取+count+个字节); System.out.print(new String(buf); fis.close(); /关闭fis输入流 catch (IOException ioe) System.out.println(I/O异常); ,FilterInputStream,FilterInputStream是为了包装InputStream流而引入的中间类,说它是中间类,是因为它的构造方法的访问属性为protected的,即用户不能直接将其实例化,创建FilterInputStream对象,它把具体的包装任务交给了它的子类们来完成。 这些子类有BufferedInputStrea

10、m、CheckedInputStream、DataInputStream、 LineNumberInputStream等,每一个子类都是以现成的InputStream流对象为其数据源,试图对该InputStream流做进一步的处理。 当然,有兴趣的读者也可以试着自己定义一个从FilterInputStream继承而来的加强输入流类,实现对输入流的特殊处理(如按位读取等)。,BufferedInputStream,BufferedInputStream类只是在FilterInputStream类(或者说InputStream类)的基础上添加了一个读取缓冲功能,因此,也有人说它本来应该合并到Inp

11、utStream中去才对。不过,我们更关心的是,到底缓冲能带来多大的性能提高呢? 例12-7就是一个测试程序,读者有兴趣的话可以亲自上机验证一下,我们在自己的计算机上对输入流的缓冲与否做了一个测试,测试读取的为一图片文件,大小约为2.52M,结果表明,它们二者之间的速度差别还是非常明显的,对于小输入流的读取况且如此,那么对于大输入流情况,则缓冲带来的效果就可想而知了。,【例 12-7】测试BufferedInputStream输入流类带来的性能提高。 import java.io.*; public class TestBufferedInputStream public static voi

12、d main(String args) throws IOException try /创建文件输入流对象fis,为了取得明显效果,Big.dat文件中编辑了大量数据 InputStream fis =new BufferedInputStream( new FileInputStream(Big.dat); System.out.println(测试开始.); while (fis.read()!=-1) /从文件输入流fis中读取字节数据 /读取整个文件输入流 System.out.println(测试结束); fis.close(); /关闭fis输入流 catch (IOExcepti

13、on ioe) System.out.println(I/O异常); ,OutputStream,抽象类OutputStream是所有字节输出流类的基类,它的派生关系如下图所示:,FileOutputStream,【例 12-8】FileOutputStream文件输出流类。 import java.io.*; public class TestFileOutputStream public static void main(String args) try System.out.print(请输入数据: ); int count,n=128; byte buffer = new byten;

14、 count = System.in.read(buffer); /读取标准输入流 FileOutputStream fos = new FileOutputStream(test.dat); /创建文件输出流对象 fos.write(buffer,0,count); /写入输出流 fos.close(); /关闭输出流 System.out.println(已将上述输入数据输出保存为test.dat文件。); catch (IOException ioe) System.out.println(ioe); catch (Exception e) System.out.println(e);

15、,12.4 字符流,字符流类是为方便处理16位Unicode字符而(在JDK1.1之后)引入的输入输出流类,它以两个字节为基本输入输出单位,适合于处理文本类型数据。 Java设计的字符流体系中有两个基本类:Reader和Writer,分别对应字符输入流和字符输出流。,12.4.1 Reader,Reader字符输入流是一个抽象类,本身不能被实例化,因此真正实现字符流输入功能的是由它派生的子类们,如BufferedReader、CharArrayReader、FilterReader、InputStreamReader、PipedReader和StringReader等,其中一些子类又再进一步派

16、生出其他功能子类,其继承关系如下图所示:,特色,Java输入输出的一个特色就是可以组成使用(包装)各种输入输出流为功能更强的流,因此,才设计定义了这么多各具功能的输入输出流类。 下面请看一个程序例子。,【例 12-9】FileReader和BufferedReader的组合使用。 import java.io.*; public class TestFileReader public static void main(String args) try FileReader fr = new FileReader(fuwa.dat); BufferedReader bfr = new Buffe

17、redReader(fr); String str=bfr.readLine(); while (str!=null) System.out.println(str); str=bfr.readLine(); catch (IOException ioe) System.out.println(ioe); catch (Exception e) System.out.println(e); ,12.4.2 Writer,字符流输出基类Writer也是一个抽象类,本身不能被实例化,因此真正实现字符流输出功能的是由它派生的子类们,如BufferedWriter、CharArrayWriter、Fi

18、lterWriter、 OutputStreamWriter、PipedWriter、PrintWriter和StringWriter等,其中OutputStreamWriter子类又再进一步派生出FileWriter子类,其继承关系如下图所示:,FileWriter类的其他方法都是从它的父类继承而来的。在实际应用中,常将FileWriter类的对象包装为BufferedWriter对象,以提高字符输出效率。 请看下面的例子。,【例 12-10】FileWriter和BufferedWriter的组合使用。 import java.io.*; public class TestFileWrit

19、er public static void main(String args) try InputStreamReader isr = new InputStreamReader(System.in); BufferedReader br = new BufferedReader(isr); FileWriter fw = new FileWriter(out.dat); BufferedWriter bw = new BufferedWriter(fw); String str = br.readLine(); while(!(str.equals(#) bw.write(str,0,str

20、.length(); bw.newLine(); str = br.readLine(); br.close(); bw.close(); catch(IOException e) e.printStackTrace(); ,注意,bw.newLine();语句在不同系统下实际输出的行分隔符是不同的,在Windows下是“r”(回车)和“n”(换行),在Unix/Linux下只有“n”,而在Mac OS下则是“r”,因此,如果在Windows下用记事本程序打开Unix/Linux下编辑的文本文件,就会看不到分行的效果,要想恢复原来的分行效果,可以通过将每一个“n”转换为“r”和“n”,这样,就

21、可以恢复Unix/Linux下分行的效果了。 下面请看实现这一转换过程的一个程序示例。,【例 12-11】Unix文本文件转换为Windows文本文件。 import java.io.*; public class Unix_2_Win public static void main(String args) try FileReader fileReader = new FileReader(unix.dat); FileWriter fileWriter = new FileWriter(win.dat); char line = r, n; int ch = fileReader.rea

22、d(); while(ch != -1) /直到文件结束 if(ch = n) fileWriter.write(line); /实施转换 else fileWriter.write(ch); /不变 ch = fileReader.read(); /读取下一个字符 fileReader.close(); /关闭输入流 fileWriter.close(); /关闭输出流 catch(IOException e) e.printStackTrace(); ,Unix下编辑的文本文件unix.dat在Windows下用记事本打开,如下图所示:,当执行上述程序,对Unix.dat文件进行读取并转换

23、保存为win.dat后,再用记事本打开,显示效果如下图所示:,PipedWriter,PipedWriter为管道字符输出流类,它必须与相应的PipedReader一起工作,共同实现管道式输入输出。,12.5 文件 12.5.1 File类,与java.io包定义的其他输入输出类不同的是,File类直接处理文件和文件系统本身,也就是说File类并不关心怎样从文件读取数据流或向文件存储数据流,它主要用来描述文件或目录的自身属性。 通过创建File类对象,我们可以处理和获取与文件相关的信息,比如文件名、相对路径、绝对路径、上级目录、是否存在、是否是目录、可读、可写、上次修改时间和文件长度等。 此外

24、,当File对象为目录时,还可以列举出它的文件和子目录。 一个File类对象被创建后,它的内容就不能再改变了,要想改变(即进行文件读写操作)就必须利用前面介绍过的强大I/O流类对其进行包装或者使用后面即将介绍的RandomAccessFile类。 总之,对于Java语言,不管是文件还是目录都用File类来表示。,【例 12-13】File类示例程序。 import java.io.*; import java.util.*; public class TestFile public static void main(String args) try File f = new File(args

25、0); if(f.isFile() / 是否是文件 System.out.println(该文件属性如下所示:); System.out.println(文件名- +f.getName(); System.out.println(f.isHidden()? -隐藏 : -没隐藏); System.out.println(f.canRead() ? -可读 : -不可读 ); System.out.println(f.canWrite() ? -可写 : -不可写 ); System.out.println(大小- +f.length() + 字节); System.out.println(最后

26、修改时间- +new Date(f.lastModified(); ,else / 列出所有的文件和子目录 File fs = f.listFiles(); ArrayList fileList = new ArrayList(); for(int i = 0; i +f.getName(); System.out.println(); catch(ArrayIndexOutOfBoundsException e) System.out.println(e.toString(); ,12.5.2 RandomAccessFile类,上述File类不能进行文件读写操作,必须通过其他类来提供该功能

27、,RandomAccessFile类就是其中之一。 RandomAccessFile类与前面介绍过的文件输入输出流类相比,其文件存取方式更灵活,它支持文件的随机存取(Random Access):在文件中可以任意移动读取位置。 RandomAccessFile类对象可以使用seek()方法来移动文件存取的位置,移动单位为字节,为了能正确移动存取位置,编程者必须清楚随机存取文件中各数据的长度和组织。,【例 12-14】RandomAccessFile类示例程序。 import java.io.*; import java.util.*; import myPackage.MyInput; /定义

28、图书类Book class Book private StringBuffer name; private short price; /2个字节 public Book(String n,int p) name=new StringBuffer(n); name.setLength(7); /限定为固定的7个字符(14字节) price=(short)p; public String getName() return name.toString(); public short getPrice() return price; public static int size() return 16

29、; ,public class TestRandomAccessFile public static void main(String args) throws IOException Book books = new Book(Java教程, 22),new Book(操作系统, 38), new Book(编译原理, 29),new Book(计算机网络, 32), new Book(计算机图形学, 18),new Book(数据库原理, 12); File f = new File(stock.dat); /以读写方式打开stock.dat文件 RandomAccessFile raf

30、= new RandomAccessFile(f, rw); /将books中的书本信息写入文件 for(int i = 0; i books.length; i+) raf.writeChars(booksi.getName(); raf.writeShort(booksi.getPrice(); ,System.out.print(查询第几本书?); /利用自定义类MyInput进行数据输入 int n = MyInput.intData(); /通过seek()定位到第n本书的数据起始位置 raf.seek(n-1) * Book.size(); /bname用于存放读取到的第n本书的书

31、名 char bname=new char7; char ch; for(int i=0;i7;i+) ch = raf.readChar(); if (ch=0) bnamei=0; else bnamei=ch; System.out.print(书名:); System.out.println(bname); System.out.println(单价: + raf.readShort(); /输出读取到的第n本书的单价 raf.close(); /关闭文件 ,提示,所有文件都是二进制文件,至于文件中的二进制数据如何解读?这完全取决于它的数据组织方式和编码格式。 总之,编码(以及相应的解码)是计算机之所以能用简单的0、1来表达整个世界的关键!,文件读写操作三步骤,一般进行文件读写操作应包括如下三个步骤: 1、以某种读写方式打开文件; 2、进行文件读写操作; 3、关闭文件。 需要特别注意的是:对于某些文件存取对象来说,关闭文件的动作就意味着将缓冲区(Buffer)中的数据全部写入磁盘文件,如果不进行(或忘记)文件关闭操作,某些数据可能就会因为没能及时写入文件而发生丢失。,作业,1 2 5 7 8 12,

展开阅读全文
相关资源
猜你喜欢
相关搜索

当前位置:首页 > 企业管理 > 管理学资料

本站链接:文库   一言   我酷   合作


客服QQ:2549714901微博号:道客多多官方知乎号:道客多多

经营许可证编号: 粤ICP备2021046453号世界地图

道客多多©版权所有2020-2025营业执照举报