java Writer的基本使用
1、
1 package cn.kongxh.io3; 2 3 import java.io.File ; 4 import java.io.Writer ; 5 import java.io.FileWriter ; 6 import java.io.OutputStreamWriter; 7 public class WriterDemo01{ 8 public static void main(String args[]) throws Exception{ // 异常抛出,不处理 9 // 第1步、使用File类找到一个文件 10 File f= new File("d:" + File.separator + "test.txt") ; // 声明File对象 11 // 第2步、通过子类实例化父类对象 12 Writer out = null ; // 准备好一个输出的对象 13 out = new FileWriter(f) ; // 通过对象多态性,进行实例化 14 // 第3步、进行写操作 15 String str = "Hello World!!!" ; // 准备一个字符串 16 out.write(str) ; // 将内容输出,保存文件 17 // 第4步、关闭输出流 18 out.close() ; // 关闭输出流 19 //public class FileWriter extends OutputStreamWriter 20 //public class OutputStreamWriter extends Writer FileWriter实际上继承自 OutputStreamWriter 21 } 22 };
2、
1 package cn.kongxh.io3; 2 3 import java.io.File ; 4 import java.io.Writer ; 5 import java.io.FileWriter ; 6 public class WriterDemo02{ 7 public static void main(String args[]) throws Exception{ // 异常抛出,不处理 8 // 第1步、使用File类找到一个文件 9 File f= new File("d:" + File.separator + "test.txt") ; // 声明File对象 10 // 第2步、通过子类实例化父类对象 11 Writer out = null ; // 准备好一个输出的对象 12 out = new FileWriter(f,true) ; // 通过对象多态性,进行实例化 13 // 第3步、进行写操作 14 String str = "\r\nLIXINGHUA\r\nHello World!!!" ; // 准备一个字符串 15 out.write(str) ; // 将内容输出,保存文件 16 // 第4步、关闭输出流 17 out.close() ; // 关闭输出流 18 } 19 };
3、
1 package cn.kongxh.io3; 2 3 import java.io.File ; 4 import java.io.Writer ; 5 import java.io.FileWriter ; 6 public class WriterDemo03{ 7 public static void main(String args[]) throws Exception{ // 异常抛出,不处理 8 // 第1步、使用File类找到一个文件 9 File f= new File("d:" + File.separator + "test.txt") ; // 声明File对象 10 // 第2步、通过子类实例化父类对象 11 Writer out = null ; // 准备好一个输出的对象 12 out = new FileWriter(f) ; // 通过对象多态性,进行实例化 13 // 第3步、进行写操作 14 String str = "Hello World!!!" ; // 准备一个字符串 15 out.write(str) ; // 将内容输出,保存文件 16 // 第4步、关闭输出流 17 // out.close() ; // 此时,没有关闭 18 /* 19 * 不关闭 不会写入到文件中 证明还在缓存区,没有刷到文件中 如果关闭流(out.close();) 会强制刷新缓存区 20 * */ 21 } 22 };
4、
1 package cn.kongxh.io3; 2 3 import java.io.File ; 4 import java.io.Writer ; 5 import java.io.FileWriter ; 6 public class WriterDemo04{ 7 public static void main(String args[]) throws Exception{ // 异常抛出,不处理 8 // 第1步、使用File类找到一个文件 9 File f= new File("d:" + File.separator + "test.txt") ; // 声明File对象 10 // 第2步、通过子类实例化父类对象 11 Writer out = null ; // 准备好一个输出的对象 12 out = new FileWriter(f) ; // 通过对象多态性,进行实例化 13 // 第3步、进行写操作 14 String str = "Hello World!!!" ; // 准备一个字符串 15 out.write(str) ; // 将内容输出,保存文件 16 // 第4步、关闭输出流 17 out.flush() ; // 强制性清空缓冲区中的内容 18 // out.close() ; // 此时,没有关闭 19 } 20 };
5、案例
1 package cn.kongxh.io3; 2 import java.io.* ; 3 public class Copy{ 4 public static void main(String args[]){ 5 if(args.length!=2){ // 判断是否是两个参数 6 System.out.println("输入的参数不正确。") ; 7 System.out.println("例:java Copy 源文件路径 目标文件路径") ; 8 System.exit(1) ; // 系统退出 9 } 10 File f1 = new File(args[0]) ; // 源文件的File对象 11 File f2 = new File(args[1]) ; // 目标文件的File对象 12 if(!f1.exists()){ 13 System.out.println("源文件不存在!") ; 14 System.exit(1) ; 15 } 16 InputStream input = null ; // 准备好输入流对象,读取源文件 17 OutputStream out = null ; // 准备好输出流对象,写入目标文件 18 try{ 19 input = new FileInputStream(f1) ; 20 }catch(FileNotFoundException e){ 21 e.printStackTrace() ; 22 } 23 try{ 24 out = new FileOutputStream(f2) ; 25 }catch(FileNotFoundException e){ 26 e.printStackTrace() ; 27 } 28 if(input!=null && out!=null){ // 判断输入或输出是否准备好 29 int temp = 0 ; 30 try{ 31 while((temp=input.read())!=-1){ // 开始拷贝 32 out.write(temp) ; // 边读边写 33 } 34 System.out.println("拷贝完成!") ; 35 }catch(IOException e){ 36 e.printStackTrace() ; 37 System.out.println("拷贝失败!") ; 38 } 39 try{ 40 input.close() ; // 关闭 41 out.close() ; // 关闭 42 }catch(IOException e){ 43 e.printStackTrace() ; 44 } 45 } 46 } 47 }
总结:
文件IO操作、字节流和字符流
无缓冲区 阻塞
实际上字节流在操作时本身不会用到缓冲区(内存),是文件本身直接操作的,而字符流在操作时使用了缓冲区,通过缓冲区再操作文件
提问:什么叫缓冲区?
在很多地方都碰到缓冲区这个名词,那么到底什么是缓冲区?又有什么作用呢?
回答:缓冲区可以简单地理解为一段内存区域。
可以简单地把缓冲区理解为一段特殊的内存。
某些情况下,如果一个程序频繁地操作一个资源(如文件或数据库),则性能会很低,此时为了提升性能,就可以将一部分数据暂时读入到内存的一块区域之中,以后直接从此区域中读取数据即可,因为读取内存速度会比较快,这样可以提升程序的性能。
在字符流的操作中,所有的字符都是在内存中形成的,在输出前会将所有的内容暂时保存在内存之中,所以使用了缓冲区暂存数据。
如果想在不关闭时也可以将字符流的内容全部输出,则可以使用Writer类中的flush()方法完成。
在回答之前,先为读者讲解这样的一个概念,所有的文件在硬盘或在传输时都是以字节的方式进行的,包括图片等都是按字节的方式存储的,而字符是只有在内存中才会形成,所以在开发中,字节流使用较为广泛。
字节流与字符流主要的区别是他们的的处理方式
流分类:
1.Java的字节流
InputStream是所有字节输入流的祖先,而OutputStream是所有字节输出流的祖先。
2.Java的字符流(字符只存在内存里)
Reader是所有读取字符串输入流的祖先,而writer是所有输出字符串的祖先。
InputStream,OutputStream,Reader,writer都是抽象类。所以不能直接new
字节流是最基本的,所有的InputStream和OutputStream的子类都是,主要用在处理二进制数据,它是按字节来处理的
但实际中很多的数据是文本,又提出了字符流的概念,它是按虚拟机的encode来处理,也就是要进行字符集的转化
这两个之间通过 InputStreamReader,OutputStreamWriter来关联,实际上是通过byte[]和String来关联
在实际开发中出现的汉字问题实际上都是在字符流和字节流之间转化不统一而造成的
在从字节流转化为字符流时,实际上就是byte[]转化为String时,
public String(byte bytes[], String charsetName)
有一个关键的参数字符集编码,通常我们都省略了,那系统就用操作系统的lang
而在字符流转化为字节流时,实际上是String转化为byte[]时,
byte[] String.getBytes(String charsetName)
也是一样的道理
至于java.io中还出现了许多其他的流,按主要是为了提高性能和使用方便,
如BufferedInputStream,PipedInputStream等