Java基础系列——IO流

---恢复内容开始---

Java对数据的操作都是通过流的方式,数据的输入和输出是相对内存来说的,将外设的数据读到内存:输入流;将内存的数据写到外设:输出流。

流按操作数据分为两种:字节流,字符流。

输入输出的数据全都是以字节为单位的二进制,字符流后期才出现,字符流出现的目的是让机器识别日常生活文字,编码表(美国ascii,中国GBK)记录了文字和数字的对应关系,从而字节流+编码表=字节流,从硬盘读取字节流之后,先查表,转换成对应的文字,从而形成字符流。(注:国际统一的为unicode(java内置),无论什么字符都用2个字节表示)

字节流的抽象基类(顶层父类):inputStream,outputStream.(数据载体)

字符流的抽象基类 :Reader,Writer.

这些子类的后缀都是父类名,前缀是子类实现的功能。

java.lang.Object                                                                                                java.lang.Object

|--java.io.writer   (写入字符流的抽象类)                                                                  |--java.io.writer

|--java.io.OutputStreamWriter(字符流通向字节流的桥梁,转换流)                                  |--BufferedWriter

|--java.io.FileWriter

1.FileWriter:将字符写入到文本文件(和文本编辑器,word等工具一个原理)

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

FileWriter fw = new FileWriter("demo.txt",true);       // 如果demo存在,则会覆盖,如果不存在,创建。如果加入      true,那么不会覆盖,会续写。

fw.write("abc");                                                     //abc写入到了临时存储缓冲区中。

fw.flush();                                                            //将数据直接写入到demo。

fw.close();                                                            //关闭流,释放资源。close中调用了flush,flush可省略。

}

IO异常的处理:

public static void main(String[] args){

FileWriter fw = null;                                             //流对象的创建方式,因为一般都会发生异常。

try{

fw = new FileWriter("demo.txt",true);

fw.write("abc");

}

catch (IOException e){

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

}

finally{

if(fw != null)                                   //如果要创建的路径不存在,比如F盘没有,那么fw这个流就不能创建,如果这里不判断,就会抛出异常。

try{

fw.close();

}   catch (IOException e) {throw new RuntimeException("关闭失败");}

}

}

2.FileReader:读取文本文件中的字符

FileReader fr = new FileReader("demo.txt");

int a = 0;

while((a=fr.read())!=-1)

{

sop(char(a));                                                          //每次读取一个字符,但是以二进制的数字格式输出(0-65535  0x00-0xffff)。如果读取完了,会返回-1

}

/*

也可以以数组的形式来读

char[] buf = new char[3];

int num = fr.read(buf);                                           //num是读取的字符个数

sop(num+":"+new String(buf,0,num));                    //这里string里用num可以保证读几个打印几个。

如果存的是abcde五个字符,那么第一次逐个读取abc到内存里的长度为3的数组,然后输出num=3和abc,下次读取时de放到了内存数组的前两位,覆盖了ab所在位置, 第三位读不到但是内存中还是c,所以第二次读取num=2和dec,第三次返回num=0和dec。

while((num=fr.read(buf))!=-1){

sop(new String(buf,0,num));                                  //buf越大,效率越高

}

*/

例子:复制文本文件 先读再写

public static void main(String[] args) {
  FileReader fr = null;
        FileWriter fw = null;
        try{
         fr = new FileReader("demo.txt");
         fw = new FileWriter("demo123.txt");
         char[] buf = new char[1024];                                                 // 读流读出来以后,放到内存中缓冲区,然后进写流,写流flush或close,写入demo123

int ch = 0;
         while((ch=fr.read())!=-1)
         {
          fw.write(buf);
         }
        }
         catch(IOException e){
          e.toString();
         }
         finally{
          if(fw!=null )
           try{
            fw.close();
            
           }
          catch( Exception e){
           throw new RuntimeException("读写异常");
          }
          if(fr!=null)
           try{
            fr.close();
           }
          catch( Exception e){
           throw new RuntimeException("读写异常");
          }
         }      
 }

3.字符流的缓冲区BufferedReader,BufferedWriter(其实就是在类里面封装的上面的char数组)

BufferedReader和BufferedWriter,结合流才可以使用,提高流的使用效率。 数据库的连接池

BufferedWriter:将文本写入字符输出流,缓冲各个字符,从而提供单个字符,数组和字符串的高效写入。可以设置大小,但是默认大小一般够用。

方法:行分隔符newline();由line.seperate定义。

public static void main(String[] args) throws IOException {
   FileWriter fw = new FileWriter("demo1.txt");

FileReader fr = new FileReader("demo.txt");

BufferedReader bufr = new BufferedReader(fr);
   BufferedWriter bufw = new BufferedWriter(fw);

String line = null;

while((line = bufr.readLine())!=null){        //readLine()是BufferedReader特有方法,其中使用了BufferedReader的read方法,将读取到的字符串缓冲并判断换行标记,然后将标记前的字符串返回。

bufw.write(line);

bufw.newLine();                            //换行     newLine()是BufferedWriter特有方法

bufw.flush();

}

bufr.close();

bufw.close();                                    //关掉的是流,不是缓冲区

}

java.lang.Object

|--java.io.OutputStream

|--java.io.FileOutputStream

4.FileOutputStream,FileInputStream,BufferedInputStream,BufferedOutputStream.流只能操作文本文件,而字节流可以操作多种。

private static void copy() {
  FileOutputStream fos = null;
  FileInputStream fis = null;
         try{
         fis = new FileInputStream("D:\\cover.jpg");
         BufferedInputStream buis = new BufferedInputStream(fis);
    
         fos = new FileOutputStream("E:\\cover.jpg");
         BufferedOutputStream buos = new BufferedOutputStream(fos);
    
        int ch = 0;
        ry {
        while((ch=buis.read(buf))!=-1){
             try{
             buos.write(buf);
           }
       catch(IOException e){
       System.out.println("失败"); 
     }
     } catch (IOException e) {
     e.printStackTrace();
     }
   
    catch (FileNotFoundException e) {
    e.printStackTrace();
 }finally{
  try {
   if(fis==null)
   fis.close();
   if(fos==null)
   fos.close();
   System.out.println("复制成功");
  } catch (IOException e) {
   e.printStackTrace();
  }
 }

5.字节流和字符流之间的转换 InputStreamReader,OutputStreamWriter

 InputStreamReader将字节流转换成字符流(读取的还是字节流,但是它拿着字节去查表去了,返回的是字符的ASCII值,一个汉字两个字节,如果不转的话,分别输出两个字节值,转换成字符流,输出一个值)         解码的过程

从 InputStream in = System.in;                                                     上 OutputStream out = System.out;
    上 InputStreamReader isr = new InputStreamReader(in);                向 OutputStreamWriter osw = new OutputStreamWriter(out);
    向 BufferedReader bufr = new BufferedReader(isr);                         下 BufferedWriter bufw = new BufferedWriter(osw);
    下 String line = bufr.readLine();               ------------------------->   从 bufw.write(line);

简化写法 BufferedReader bufr = new BufferedReader(new InputStreamReader(System.in或者是FileInputStream("demo.txt")));

BufferedWriter bufw = new BufferedWriter(new OutputStreamWriter(System.out或者是FileOutputStream("demo.txt")));

注意点:FileWriter fw = new FileWriter("demo.txt");效果等同于

OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("demo.txt"),"GBK");

FileWriter是子类,为操作文件而生,默认GBK,其中封装了转换流功能,虽然看起来是直接操作文本中的字符,但说到底文本还是要从硬盘读取,硬盘中的数据是字节。

6.File类

File f1 = new File("c:\\a.txt");   File f2 = new File("c:\\","a.txt");

File f = new File("c:\\");      File f4 = new File(f,"a.txt");

File f = new File("c:"+File.separator+"睿思"+File.separator+"a.txt");   分隔符File.separator

具体方法:文件对象的状态获取,创建删除,判断,重命名,获取根目录和容量,目录内容,过滤指定类型文件的过滤器(FilenameFilter接口)等。

7.Properties类(与流相关)

Map
   |--HashTable
      |--Properties
1.集合中的键和值都是字符串类型。
2.该集合中的数据可以保存到流中,或者从流获取。操作以键值对形式存在的配置文件。

功能:将配置文件以流的形式读出,然后利用properties的方法对其中的键值形式信息进行操作。

1.Properties p = new Properties();
     p.setProperty("zhangsan", "22");
     FileOutputStream fos = new FileOutputStream("demo.txt");
     p.store(fos, "name+age");
     fos.close();

2. File file = new File("demo.txt");
      if(!file.exists()){
      file.createNewFile();
      }
      FileReader fr = new FileReader(file);
      Properties p = new Properties();
      p.load(fr);
      p.list(System.out);//列出文件内容,测试是否关联成功
      p.setProperty("wangwu", "33");
      FileWriter fw = new FileWriter(file);
      fw.close();
      fr.close();

3.记录应用程序打开次数

File confile = new File("confile.txt");
   if(!confile.exists()){
    confile.createNewFile();
   }
   FileReader fr = new FileReader(confile);
   Properties p = new Properties();
   p.load(fr);
   String name = p.getProperty("time");
   int value = 0;
   if(name!=null){
    value = Integer.parseInt(name);
   }
   value++;
   p.setProperty("time", "value");

4.返回目录下文件名以.java结尾的文件(附件

public class Test {

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

          File dir = new File("c:\\");
FilenameFilter filter = new FilenameFilter(){
public boolean accept(File dir,String name){
return name.endsWith(".java");
}
};
List<File> list = new ArrayList<File>();
getFiles(dir,filter,list);
File destFile = new File(dir,"javalist.txt");
write2File(list,destFile);
} public static void getFiles(File dir,FilenameFilter filter,List<File> list){
File[] files = dir.listFiles();
for(File file : files){
if(file.isDirectory()){
getFiles(file,filter,list);
}
else{
if(filter.accept(dir, file.getName()));{
list.add(file);
}
}
}
}
public static void write2File(List<File> list,File destFile){
BufferedWriter bufw = null;
try {
bufw = new BufferedWriter(new FileWriter(destFile));
for(File file:list){
bufw.write(file.getAbsolutePath());
bufw.newLine();
bufw.flush();
}
} catch (IOException e) {
throw new RuntimeException();
}
finally{
if(bufw!=null){
try{
bufw.close();
}catch(IOException e){
throw new RuntimeException("关闭失败");
}
}
}
} }

8.打印流 PrintStream PrintWriter

可以对多种数据类型值进行打印,并保持数据的表示形式。不抛IOException。想保证数据形式原样时,用print方法。

PrintStream 打印的所有字符都使用平台的默认字符编码转换为字节。在需要写入字符而不是写入字节的情况下,应该使用
PrintWriter类。

构造函数接受三种类型值:字符串路径,File对象,字节输出流

PrintStream out = new PrintStream("print.txt");

out.print(97);

out.close();

      9.序列流SequenceInputStream

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

           ArrayList<FileInputStream> al = new ArrayList<FileInputStream>();
al.add(new FileInputStream("demo1.txt"));al.add(new FileInputStream("demo2.txt"));al.add(new FileInputStream("demo3.txt"));
Enumeration<FileInputStream> en = Collection.enumeration(al); //迭代器
SequenceInputStream sis = new SequenceInputStream(en);
FileOutputStream fos = new FileOutputStream("demo.txt");
byte[] buf = new byte[1024];
int len = 0;
while((len = sis.read(buf))!=-1){
fos.write(buf,0,len);
}
fos.close();
sis.close();
}
上一篇:人生苦短之Python的urllib urllib2 requests


下一篇:HDU-5391 Zball in Tina Town