文件输入输出流

文件输入输出流 FileInputStream 和 FileOutputStream 负责完成对本地磁盘文件的顺序输入输出操作。

【例 10-5】通过程序创建一个文件,从键盘输入字符,当遇到字符“#”时结束,在屏幕上显示该文件的所有内容。

 【例 10-5】通过程序创建一个文件,从键盘输入字符,当遇到字符“#”时结束,在屏幕上显示该文件的所有内容。
//********** ep10_5.java **********
import java.io.*;
class ep10_5{
public static void main(String args[]){
char ch;
int data;
try{
FileInputStream a=new FileInputStream(FileDescriptor.in); //创建文件输入流对象
FileOutputStream b=new FileOutputStream("ep10_5"); //创建文件输出流对象
System.out.println("请输入字符,以#号结束:");
while((ch=(char)a.read())!='#'){
b.write(ch);
}
a.close();
b.close();
System.out.println();
FileInputStream c=new FileInputStream("ep10_5");
FileOutputStream d=new FileOutputStream(FileDescriptor.out);
while(c.available()>0){
data=c.read();
d.write(data);
}
c.close();d.close();
}
catch(FileNotFoundException e){
System.out.println("找不到该文件!");
}
catch(IOException e){}
}
}

运行后在程序目录建立一个名称为 ep10_5 的文件,运行结果如图 10-8 所示:


图 10-8  例 10_5 运行结果

FileDescriptor 是 java.io 中的一个类,该类不能实例化,其中包含三个静态成员:in、out 和err,分别对应于标准输入流、标准输出流和标准错误流,利用它们可以在标准输入输出流上建立文件输入输出流,实现键盘输入或屏幕输出操作。

【例 10-6】实现对二进制图形文件(.gif)的备份。

 【例 10-6】实现对二进制图形文件(.gif)的备份。
//********** ep10_6.java **********
import java.io.*;
class ep10_6{
public static void main(String args[]) throws IOException{
FileInputStream a=new FileInputStream("ep10_6.gif");
FileOutputStream b=new FileOutputStream("ep10_6_a.gif");
System.out.println("文件的大小为:"+a.available());
byte c[]=new byte[a.available()];
a.read(c); //将图形文件读入数组
b.write(c); //将数组中的数据写入到新文件中
System.out.println("文件已经被更名复制!");
a.close();
b.close();
}
}

运行后在程序目录备份了一个名称为 ep10_6_a.gif 的文件,运行结果如图 10-9 所示:


图 10-9  例 10_6 运行结果

过滤流

FilterInputStream 和 FileOutputStream 是 InputStream 和 OutputStream 的直接子类,分别实现了在数据的读、写操作的同时能对所传输的数据做指定类型或格式的转换,即可实现对二进制字节数据的理解和编码转换。

常用的两个过滤流是数据输入流 DataInputStream 和数据输出流 DataOutputStream。其构造方法为:
    DataInputStream(InputStream in);  //创建新输入流,从指定的输入流 in 读数据
    DataOutputStream(OutputStream out);  //创建新输出流,向指定的输出流 out 写数据

由于 DataInputStream 和 DataOutputStream 分别实现了 DataInput 和 DataOutput 两个接口(这两个接口规定了基本类型数据的输入输出方法)中定义的独立于具体机器的带格式的读写操作,从而实现了对不同类型数据的读写。由构造方法可以看出,输入输出流分别作为数据输入输出流的构造方法参数,即作为过滤流必须与相应的数据流相连。

DataInputStream 和 DataOutputStream 类提供了很多个针对不同类型数据的读写方法,具体内容读者可参看 Java 的帮助文档。

【例 10-7】将三个 int 型数字 100,0,-100 写入数据文件 ep10_6.dat 中。

 【例 10-7】将三个 int 型数字 100,0,-100 写入数据文件 ep10_6.dat 中。
//********** ep10_7.java **********
import java.io.*;
class ep10_7{
public static void main(String args[]){
String fileName="ep10_7.dat";
int value1=100,value2=0,value3=-100;
try{
//将 DataOutputStream 与 FileOutputStream 连接输出不同类型的数据
DataOutputStream a=new DataOutputStream(new FileOutputStream(fileName));
a.writeInt(value1);
a.writeInt(value2);
a.writeInt(value3);
a.close();
}
catch(IOException i){
System.out.println("出现错误!"+fileName);
}
}
}

运行后在程序目录中生成数据文件 ep10_7.dat,用文本编辑器打开后发现内容为二进制的:

00 00 00 64 00 00 00 00 FF FF FF 9C。

【例 10-8】读取数据文件 ep10_6.dat 中的三个 int 型数字,求和并显示。

 【例 10-8】读取数据文件 ep10_6.dat 中的三个 int 型数字,求和并显示。
//********** ep10_8.java **********
import java.io.*;
class ep10_8{
public static void main(String args[]){
String fileName="D:\\myjava/ep10_7.dat";
int sum=0;
try{
DataInputStream a=new DataInputStream(new BufferedInputStream(new FileInputStream(fileName)));
sum+=a.readInt();
sum+=a.readInt();
sum+=a.readInt();
System.out.println("三个数的和为:"+sum);
a.close();
}
catch(IOException e){
System.out.println("出现错误!"+fileName);
}
}
}

运行结果:

三个数的和为:0

readInt 方法可以从输入输出流中读入 4 个字节并将其作为 int 型数据直接参与运算。由于已经知道文件中有 3 个数据,所以可以使用 3 个读入语句,但若只知道文件中是 int 型数据而不知道数据的个数时该怎么办呢?因为 DataInputStream 的读入操作如遇到文件结尾就会抛出 EOFException 异常,所以可将读操作放入 try 中。

 try{
while(true)
sum+=a.readInt();
}
catch(EOFException e){
System.out.pritnln("三个数的和为:"+sum);
a.close();
}

EOFException 是 IOException 的子类,只有文件结束异常时才会被捕捉到,但如果没有读到文件结尾,在读取过程中出现异常就属于 IOException。

【例 10-9】从键盘输入一个整数,求该数的各位数字之和。

 【例 10-9】从键盘输入一个整数,求该数的各位数字之和。
//********** ep10_9.java **********
import java.io.*;
class ep10_9{
public static void main(String args[]) throws IOException{
DataInputStream a=new DataInputStream(System.in);
System.out.print("请输入一个整数:");
int b=a.readInt();
int sum=0;
int c=b;
while(c>0){
int d=c%10; //取最低位
c=c/10; //去掉最低位
sum=sum+d; //累加各位之和
}
System.out.println(b+"的各位数字之和="+sum);
}
}

运行结果:

请输入一个整数:26
842403082 的各位数字之和=31

需要注意的是,输入的数据 26 为变成了 842403082,原因在于输入数据不符合基本类型数据的格式,从键盘提供的数据是字符的字节码表示方式,若输入 26,只代表 2 和 6 两个字符的字节数据,而不是代表整数 26 的字节码。

若要从键盘得到整数需要先读取字符串,再利用其他方法将字符串转化为整数。

标准输入输出

System.in、System.out、System.err 这 3 个标准输入输流对象定义在 java.lang.System 包中,这 3 个对象在 Java 源程序编译时会被自动加载。

  1. 标准输入:标准输入 System.in 是 BufferedInputStream 类的对象,当程序需要从键盘上读入数据时,只需要调用 System.in 的 read()方法即可,该方法从键盘缓冲区读入一个字节的二进制数据,返回以此字节为低位字节,高位字节为 0 的整型数据。
  2. 标准输出:标准输出 System.out 是打印输出流 PrintStream 类的对象。PrintStream 类是过滤输出流类 FilterOutputStream 的一个子类,其中定义了向屏幕输出不同类型数据的方法print()和 println()。
  3. 标准错误输出:System.err 用于为用户显示错误信息,也是由 PrintStream 类派生出来的错误流。Err 流的作用是使 print()和 println()将信息输出到 err 流并显示在屏幕上,以方便用户使用和调试程序。

【例 10-10】输入一串字符显示出来,并显示 System.in 和 System.out 所属的类。

 【例 10-10】输入一串字符显示出来,并显示 System.in 和 System.out 所属的类。
//********** ep10_10.java **********
import java.io.*;
class ep10_10{
public static void main(String args[]){
try{
byte a[]=new byte[128]; //设置输入缓冲区
System.out.print("请输入字符串:");
int count =System.in.read(a); //读取标准输入输出流
System.out.println("输入的是:");
for(int i=0;i<count;i++)
System.out.print(a[i]+""); //输出数组元素的 ASCII 值
System.out.println();
for(int i=0;i<count-2;i++) //不显示回车和换行符
System.out.print((char)a[i]+""); //按字符方式输出元素
System.out.println();
System.out.println("输入的字符个数为:"+count);
Class InClass=System.in.getClass();
Class OutClass=System.out.getClass();
System.out.println("in 所在的类为:"+InClass.toString());
System.out.println("out 所在的类为:"+OutClass.toString());
}
catch(IOException e){}
}
}

运行结果如图 10-10 所示:


图 10-10  例 10_10 运行结果

需要注意的是,输入了 3 个字符按回车后,输出的结果显示为 5 个字符。这是由于 Java 中回车被当作两个字符,一个是 ASCⅡ为 13 的回车符,一个是值为 10 的换行符。程序中 getClass()和 ToString()是 Object 类的方法,作用分别是返回当前对象所对应的类和返回当前对象的字符串表示。

系列文章:

Java知多少(70)面向字节流的应用的更多相关文章

  1. Java知多少(69)面向字节的输入输出流

    字节流以字节为传输单位,用来读写8位的数据,除了能够处理纯文本文件之外,还能用来处理二进制文件的数据.InputStream类和OutputStream类是所有字节流的父类. InputStream类 ...

  2. Java知多少(67)面向字符的输入流

    字符流是针对字符数据的特点进行过优化的,因而提供一些面向字符的有用特性,字符流的源或目标通常是文本文件. Reader和Writer是java.io包中所有字符流的父类.由于它们都是抽象类,所以应使用 ...

  3. Java知多少(68)面向字符的输出流

    面向字符的输出流都是类 Writer 的子类,其类层次结构如图 10-5 所示. 图10-5 Writer的类层次结构图 表 10-3 列出了 Writer 的主要子类及说明. 表 10-3 Writ ...

  4. Java知多少(完结篇)

    Java知多少(1)语言概述 Java知多少(2)虚拟机(JVM)以及跨平台原理 Java知多少(3) 就业方向 Java知多少(4)J2SE.J2EE.J2ME的区别 Java知多少(5) Java ...

  5. Java知多少(66)输入输出(IO)和流的概述

    输入输出(I/O)是指程序与外部设备或其他计算机进行交互的操作.几乎所有的程序都具有输入与输出操作,如从键盘上读取数据,从本地或网络上的文件读取数据或写入数据等.通过输入和输出操作可以从外界接收信息, ...

  6. Java知多少(中)

    Java知多少(上) )interface接口 )接口和抽象类的区别 )泛型详解 )泛型通配符和类型参数的范围 )异常处理基础 )异常类型 )未被捕获的异常 )try和catch的使用 )多重catc ...

  7. Java知多少(105)套接字(Socket)

    网络应用模式主要有: 主机/终端模式:集中计算,集中管理: 客户机/服务器(Client/Server,简称C/S)模式:分布计算,分布管理: 浏览器/服务器模式:利用Internet跨平台. www ...

  8. Java知多少(87)选择框和单选按钮(转)

    选择框.单选框和单选按钮都是选择组件,选择组件有两种状态,一种是选中(on),另一种是未选中(off),它们提供一种简单的 “on/off”选择功能,让用户在一组选择项目中作选择. 选择框 选择框(J ...

  9. Java知IO

    ---恢复内容开始--- Java将IO(文件.网络.终端)封装成非常多的类,看似繁杂,其实每个类的具有独特的功能. 按照存取的对象是二进制还是文本,java使用字节流和字符流实现IO. 流是java ...

随机推荐

  1. JavaScript函数使用技巧

    JavaScript中的函数是整个语言中最有趣的一部分,它们强大而且灵活.接下来,我们来讨论JavaScript中函数的一些常用技巧: 一.函数绑定 函数绑定是指创建一个函数,可以在特定的this环境 ...

  2. BZOJ.2707.[SDOI2012]走迷宫(期望 Tarjan 高斯消元)

    题目链接 一个点到达终点的期望步数 \(E_i=\sum_{(i,j)\in G}\frac{E_j+1}{out[i]}\),\(out[i]\)为点\(i\)的出度. 那么对于一个DAG可以直接在 ...

  3. 5410 ACM 杭电 01+完全背包

    题目:http://acm.hdu.edu.cn/showproblem.php?pid=5410 虽然是英文题目:但还是很好理解的.明显的背包问题 思路:如果你能想到把题目拆分成小问题,就会简单许多 ...

  4. 潭州课堂25班:Ph201805201 爬虫基础 第四课 Requests (课堂笔记)

    优雅到骨子里的Requests   1528811134432   简介   上一篇文章介绍了Python的网络请求库urllib和urllib3的使用方法,那么,作为同样是网络请求库的Request ...

  5. angularJS常用的内置方法

    angular.lowercase(aaa); angular.uppercase(bbb); angular.isString(); angular.isNumber(); angular.isAr ...

  6. struts2中的session、request 、和action往页面中传值的方法

    ActionContext.getContext().put("list", list); ActionContext.getContext().getValueStack().p ...

  7. Java API概述

    collection of APIs(Application Programming Interface) java.lang — automatically imported into Java p ...

  8. Knockout.Js官网学习(Mapping高级用法二)

    使用ignore忽略不需要map的属性 如果在map的时候,你想忽略一些属性,你可以使用ignore累声明需要忽略的属性名称集合: " }; var mapping = { 'ignore' ...

  9. Asp.net Core中使用Redis 来保存Session, 读取配置文件

    今天 无意看到Asp.net Core中使用Session ,首先要使用Session就必须添加Microsoft.AspNetCore.Session包,默认Session是只能存去字节,所以如果你 ...

  10. Android添加全屏启动画面

    有的Android软件需要在启动的时候显示一个启动画面,可以是一张图或者一些设置什么呢,还有一个好处就是,可以趁机在后台加载数据.创建启动画面一般有两种方式:1.建立一个activity,展示启动画面 ...