Java IO中---字节流与字符流 - 代码改变世界,积累成就自己 - ITeye技术网站
前面我主要是对IO进行了一个整体上的认识和对File类的相关操作进行了简单的说明。下面主要再来说说Java IO中的字节流和字符流;
在整个IO包中,流的操作包括了两种-----字符流和字节流;
字节流:
字节输入流(InputStream):
public
abstract class InputStream extends Object implements Closeable字节输出流(OutputStream):
public
abstract class OutputStream extends Object implements Closeable, Flushable
字符流:(一个字符为两个字节)
字符输入流(Reader):
public
abstract class Reader extends Object implements Readable, Closeable字符输出流(Writer):
public
abstract class Writer extends Object implements Appendable, Closeable,
Flushable
在Java中使用IO操作的步骤:
使用File找到一个文件;
使用字符流或字节流的子类为OutputStream、InputStream、Writer、Reader进行实例化操作;
进行读或写操作;
关闭(Close()),在流的操作中必须进行关闭,但BufferedReader例外,我试过;
范例:OutputStream 通过输出流写入内容
- package com.iflytek.io02;
- import java.io.File;
- import java.io.FileOutputStream;
- import java.io.OutputStream;
- /**
- * @author xudongwang 2012-1-8
- *
- * Email:[email protected]
- */
- public class OutputStreamDemo01 {
- public static void main(String[] args) throws Exception {
- File file = new File("d:" + File.separator + "demo.txt");// 要操作的文件
- OutputStream out = null;// 声明字节输出流
- // out = new FileOutputStream(file);// 通过子类实例化
- out = new FileOutputStream(file, true);// 表示可以追加j
- String str = "王旭东\r\n";// 要输出的信息
- byte b[] = str.getBytes();// 将String变为byte数组
- out.write(b);// 写入数据
- // for (int i = 0; i < b.length; i++) {
- // out.write(b[i]);
- // }// 通过循环一个个写入
- out.close();
- }
- }
范例:InputStream 通过输入流来读取内容
- package com.iflytek.io02;
- import java.io.File;
- import java.io.FileInputStream;
- import java.io.InputStream;
- /**
- * @author xudongwang 2012-1-8
- *
- * Email:[email protected]
- */
- public class InputStreamDemo01 {
- public static void main(String args[]) throws Exception {
- File file = new File("d:" + File.separator + "demo.txt");// 要操作的文件
- InputStream input = null;// 声明字节输入流
- input = new FileInputStream(file);// 通过子类进行实例化操作
- byte b[] = new byte[1024];// 开辟空间存储读取的内容
- int len = input.read(b);// 将内容读取到byte数组中
- System.out.println(new String(b, 0, len));// 输出内容
- input.close();
- }
- }
但是读取是一种常见的读取方式,但是上面的代码读取受到开辟空间的限制,所以如果现在想动态的开辟数组空间,则可以根据文件的大小来决定,采用read()方法一个个的读取数据;
范例:read()
- package com.iflytek.io02;
- import java.io.File;
- import java.io.FileInputStream;
- import java.io.InputStream;
- /**
- * @author xudongwang 2012-1-8
- *
- * Email:[email protected]
- */
- public class InputStreamDemo02 {
- public static void main(String args[]) throws Exception {
- File file = new File("d:" + File.separator + "demo.txt");
- InputStream input = new FileInputStream(file);
- byte b[] = new byte[(int) file.length()];// 开辟空间接收读取的内容
- for (int i = 0; i < b.length; i++) {
- b[i] = (byte) input.read();// 一个个的读取数据
- }
- System.out.println(new String(b));// 输入内容,直接转换
- input.close();// 关闭
- }
- }
范例:Writer
- package com.iflytek.io02;
- import java.io.File;
- import java.io.FileWriter;
- import java.io.IOException;
- import java.io.Writer;
- /**
- * @author xudongwang 2012-1-8
- *
- * Email:[email protected]
- */
- public class WriterDemo {
- public static void main(String args[]) throws IOException {
- File file = new File("d:" + File.separator + "demo.txt");
- Writer out = null;// 声明字符输出流
- // out = new FileWriter(file);// 通过子类实例化
- out = new FileWriter(file,true);//表示可以追加
- String str = "王旭东\r\n";// 要输出的信息
- out.write(str);// 写入数据
- out.close();
- }
- }
范例:Reader
- package com.iflytek.io02;
- import java.io.File;
- import java.io.FileReader;
- import java.io.IOException;
- import java.io.Reader;
- /**
- * @author xudongwang 2012-1-8
- *
- * Email:[email protected]
- */
- public class ReaderDemo {
- public static void main(String args[]) throws IOException {
- File file = new File("d:" + File.separator + "demo.txt");
- Reader input = null;
- input = new FileReader(file);
- char c[] = new char[1024];
- int len = input.read(c);
- System.out.println(new String(c, 0, len));
- input.close();
- }
- }
或者通过一个个读取:
- package com.iflytek.io02;
- import java.io.File;
- import java.io.FileReader;
- import java.io.IOException;
- import java.io.Reader;
- /**
- * @author xudongwang 2012-1-8
- *
- * Email:[email protected]
- */
- public class ReaderDemo02 {
- public static void main(String args[]) throws IOException {
- File file = new File("d:" + File.separator + "demo.txt");
- Reader input = null;
- input = new FileReader(file);
- char c[] = new char[(int) file.length()];
- for (int i = 0; i < c.length; i++) {
- c[i] = (char) input.read();// 一个个读取
- }
- System.out.println(new String(c));
- input.close();
- }
- }
下面我们来通过一组实例来观察字节流与字符流的区别:此程序就是向文件中保存内容
范例:字节流OutputStream实现:
- package com.iflytek.io02;
- import java.io.File;
- import java.io.FileOutputStream;
- import java.io.IOException;
- import java.io.OutputStream;
- /**
- * @author xudongwang 2012-1-8
- *
- * Email:[email protected]
- */
- public class OutputStreamDemo {
- public static void main(String args[]) throws IOException {
- File file = new File("d:" + File.separator + "demo.txt");
- OutputStream out = new FileOutputStream(file);
- String str = "java";
- byte b[] = str.getBytes();
- out.write(b);
- }
- }
注意:以上的操作并没有进行关闭操作,发现内容却可以正常的输出;
范例:字符流Writer实现
- package com.iflytek.io02;
- import java.io.File;
- import java.io.FileWriter;
- import java.io.IOException;
- import java.io.Writer;
- /**
- * @author xudongwang 2012-1-8
- *
- * Email:[email protected]
- */
- public class WriterDemo02 {
- public static void main(String args[]) throws IOException{
- File file = new File("d:" + File.separator + "demo.txt");
- Writer out = new FileWriter(file);
- String str = "java";
- out.write(str);
- }
- }
此时,字符流中也没有进行流的关闭操作,但与字节流不同的是,字符流并没有将内容成功的写入进去。如果此时在字符流操作的最后加入out.flush()操作,则可以成功的写入;
此时来看可以知道因为早期只有字节流,换而言之,也就是说早期是没有刷新操作的,但是因为使用了关闭,所以表示会强制刷新,注意的是其中刷新的是缓冲区(内存),但是这里我认为任何流只要初始化了最后都将其关闭了,这是一种习惯,注此处除了BufferedReader;
此时可以得到这样的一个结论:
字节流在操作的时候是直接与文件本身进行关联的,不使用缓冲区
· 字节文件 字节流直接实现底层的IO操作;
字符流在操作的时候是通过缓冲区与文件进行操作的,使用缓冲区
· 字符缓冲文件
综上比较来说,在传输或者在硬盘上保存的内容都是以字节的形式存在,所以字节流操作较多,但是在操作文件的时候字符流比较好使;
内存操作流
其主要是保存临时文件,通过内存操作流输入输出的目标是内存。使用ByteArrayInputStream(从程序向内存写入)和ByteArrayOutputStream(从内存向程序输出)完成内存的操作流。在内存操作流中所有的输入和输出都是以内存为操作源头的;
范例:通过内存流完成字符串小写变为大写的操作;
- package com.iflytek.io02;
- import java.io.ByteArrayInputStream;
- import java.io.ByteArrayOutputStream;
- /**
- * @author xudongwang 2012-1-8
- *
- * Email:[email protected]
- */
- public class ByteArrayInputStreamDemo {
- public static void main(String args[]) {
- String str = "hello world!";
- ByteArrayInputStream bis = null;// 内存输入流,从程序向内存写入
- ByteArrayOutputStream bos = null;// 内存输出流
- bis = new ByteArrayInputStream(str.getBytes());// 将内存保存在内存中
- bos = new ByteArrayOutputStream();
- int temp = 0;
- while ((temp = bis.read()) != -1) {// 依次读取
- char c = (char) temp;
- bos.write(Character.toUpperCase(c));// 输出
- }
- String newStr = bos.toString();// 取出内存输出的内容
- System.out.println(newStr);
- }
- }
注意,虽然内存流在J2EE阶段没什么感觉,但是在Java WEB中的Ajax技术中会结合XML解析和JavaScript、Ajax完成一些动态效果;
管道流:
管道流就是进行两个线程间通讯,使用PipedInputStream和PipedOutputStream两个类完成,这两个类在使用上基本上跟InputStream和OutputStream类似,唯一的区别在于连接管道的操作上public void connect(PipedOutputStream src) throws IOException,此处就暂时不举例了;
打印流:
PrintStream和PrintWriter
public
class PrintStream extends FilterOutputStream implements Appendable, Closeable从上面可知,PrintStream是OutputStream的子类,而实际上,PrintStream类属于装饰,也就是说根据实例化PrintStream类对象的不同,输出的位置也不同;
范例:使用PrintStream向文件输出
- package com.iflytek.io02;
- import java.io.File;
- import java.io.FileOutputStream;
- import java.io.PrintStream;
- /**
- * @author xudongwang 2012-1-8
- *
- * Email:[email protected]
- */
- public class PrintStreamDemo {
- public static void main(String args[]) throws Exception {
- File file = new File("d:" + File.separator + "demo.txt");
- PrintStream ps = new PrintStream(new FileOutputStream(file));
- ps.print("hello");
- ps.print("world");
- ps.print(10);
- ps.print(10.0);
- ps.close();
- }
- }
此处,发现使用打印流输出比较方便,所以建议以后都使用打印流完成输出;
在JDK1.5之后打印流可以使用格式化输出
public
PrintStream format(String format,Object... args)
- package com.iflytek.io02;
- import java.io.File;
- import java.io.FileOutputStream;
- import java.io.PrintStream;
- /**
- * @author xudongwang 2012-1-8
- *
- * Email:[email protected]
- */
- public class PrintStreamFormatDemo {
- public static void main(String args[]) throws Exception {
- File file = new File("d:" + File.separator + "demo.txt");
- PrintStream out = new PrintStream(new FileOutputStream(file));
- String name = "王旭东";
- int age = 21;
- float score = 99.0f;
- char sex = 'N';
- out.printf("姓名:%s;年龄:%d;成绩:%5.2f;性别:%c。",name,age,score,sex);
- out.close();
- }
- }
这里需要注意,在打印流中始终记住一下原则:
· 根据实例化其子类的不同,完成的打印输出功能也不同;
System对IO的支持:
System.in:它是PrintStream的实例,常用的方法就是向屏幕上打印信息,当然如果使用System.out的话也可为OutputStream实例化
System.err:表示错误的输出
- package com.iflytek.io02;
- /**
- * @author xudongwang 2012-1-8
- *
- * Email:[email protected]
- */
- public class SystemerrDemo {
- public static void main(String args[]) {
- try {
- Integer.parseInt("hello");
- } catch (Exception e) {
- System.out.println(e);//e打印出来是黑字
- System.err.println(e);//e打印出来是红字
- }
- }
- }
其区别只能从概念上讲:
System.out:一般的信息是愿意展示给用户看见的;
System.err:一般的信息是不愿意展示给用户看见的;
System.in:表示键盘的输入;
- package com.iflytek.io02;
- import java.io.InputStream;
- /**
- * @author xudongwang 2012-1-8
- *
- * Email:[email protected]
- */
- public class SystemInDemo {
- public static void main(String args[]) throws Exception {
- InputStream input = System.in;
- System.out.println("请输入内容:");
- int temp = 0;
- StringBuilder bud = new StringBuilder();
- while ((temp = input.read()) != -1) {
- char c = (char) temp;
- if (c == '\n') {
- break;
- }
- bud.append(c);
- }
- System.out.println("输入的内容是:" + bud);
- }
- }
虽然此时读取时没有长度但此处注意的是每次读取的是字节,所以在输入中文时会出现错误,只能通过后面的BufferedReader来实现;
重定向问题:
· System.out、System.err都有固定的输出目标,都是屏幕。而System.in有固定的输入目标,都是键盘;但是在System类中提供了一系列的输入输出重定向,可以改变System.out、System.er、System.in的输入输出位置。
· System. out重定向:public static void setOut(PrintStream out)
· System.err重定向:public static void setErr(PrintStream err)
· System.in重定向:public static void setIn(InputStream in)
BufferedReader:
实际上表示缓冲区读取,可以一次性的将内容全部读取出来
其构造方法:public BufferedReader(Reader in)
注意:在Java中提供了两个专门的类,去实现字节流与字符流的转换:
· InputStreamReader:将字节的输入流变为字符输入流;
文件字节流(输入流)字符流(输入流)控制台
· OutputStreamWriter:将字符的输出流变为字节的输出流;
程序字符流(输出流)字节流(输出流)文件
通过public String readLine()throws IOException来读取数据,表示一次性读取一行数据,而且这里需要注意的是如果返回的内容是String是最好的操作;