JavaIO流中对数据的操作尤为重要,掌握了基本的拷贝操作,才能将各种数据源的操作联系起来。
先来看看对文件夹的拷贝吧:
1 /** 2 * 利用递归实现文件夹的拷贝操作 3 * 分析:判断 4 * 是文件:调用拷贝文件的方法fileCopy(...) 5 * 是文件夹:创建文件夹,并使用递归实现子文件夹/子文件的判断及操作 6 * @param src:要拷贝的文件夹源头 7 * @param dest:要拷贝到的文件夹源头 8 */ 9 public static void dirCopy(File src,File dest) { 10 if(src.isFile()) { //是文件 11 fileCopy(src, dest); 12 }else { //是文件夹 13 dest.mkdirs(); 14 for(File subSrc:src.listFiles()) { //遍历子文件夹/子文件 15 dirCopy(subSrc, new File(dest,subSrc.getName())); 16 } 17 } 18 }
对文件的拷贝,我们可以这样写:
1 /** 2 * 实现文件的拷贝 3 * 输入流读取的同时输出流进行写出 4 * @param src:要拷贝的文件源头 5 * @param dest:要拷贝到的文件源头 6 */ 7 public static void fileCopy(File src,File dest) { 8 //1.创建源 9 //2.选择流 10 InputStream is = null; 11 OutputStream os = null; 12 try { 13 is = new FileInputStream(src); 14 os = new FileOutputStream(dest); 15 //3.操作:分段读取并写出 16 int len; //接收长度 17 byte[] flush = new byte[1024]; //缓冲容器,一次读写1k 18 while((len=is.read(flush))!=-1) { 19 os.write(flush, 0, len); 20 } 21 os.flush(); //写完手动刷新,避免数据在缓冲容器中(当然当流关闭时会自动刷新) 22 }catch(FileNotFoundException e) { 23 e.printStackTrace(); 24 }catch(IOException e) { 25 e.printStackTrace(); 26 }finally { 27 //4.关闭流,分别关闭,先打开的后关闭 28 try { 29 if(os!=null) { //判断是否为空,避免空指针异常 30 os.close(); 31 } 32 }catch(IOException e) { 33 e.printStackTrace(); 34 } 35 try { 36 if(is!=null) { //判断是否为空,避免空指针异常 37 is.close(); 38 } 39 }catch(IOException e) { 40 e.printStackTrace(); 41 } 42 } 43 }
以上代码只能实现对文件的拷贝操作,当然适合于拷贝任何格式的数据文件,包括视频、音频、图片等等。但是如果我想将一张图片拷贝到字节数组中呢(这里的字节数组相当于内存),也就是说从文件到字节数组,或着是从字节数组到文件。那么以上代码就具有局限性了,也不易于扩展,来看以下代码:
1 /** 2 * 对接流 3 * @param is:输入流 4 * @param os:输出流 5 */ 6 public static void copy(InputStream is,OutputStream os) { 7 //1.创建源 8 //2.选择流 9 try { 10 //3.操作 11 byte[] flush = new byte[1024]; //缓冲容器 12 int len; //接收长度 13 while((len=is.read(flush))!=-1) { 14 os.write(flush, 0, len); 15 } 16 os.flush(); 17 }catch(FileNotFoundException e) { 18 e.printStackTrace(); 19 }catch(IOException e) { 20 e.printStackTrace(); 21 }finally { 22 try { 23 if(os!=null) { 24 os.close(); 25 } 26 }catch(IOException e) { 27 e.printStackTrace(); 28 } 29 try { 30 if(is!=null) { 31 is.close(); 32 } 33 }catch(IOException e) { 34 e.printStackTrace(); 35 } 36 } 37 }
嗯,这样就可以实现以上要求了,但是我们发现:关闭资源的操作一直在,而且都一样,我们可以封装一下,这样在finally中就可以直接调用了。
1 /** 2 * 关闭的方法 3 * @param is:输入流 4 * @param os:输出流 5 */ 6 public static void close(InputStream is,OutputStream os) { 7 try { 8 if(os!=null) { 9 os.close(); 10 } 11 }catch(IOException e) { 12 e.printStackTrace(); 13 } 14 try { 15 if(is!=null) { 16 is.close(); 17 } 18 }catch(IOException e) { 19 e.printStackTrace(); 20 } 21 }
好了,看看封装的代码,幸亏只有两个流,要是流很多咋办,形参太多,但是我们发现输入流InputStream和输出流OutputStream都实现了同一个接口:Closeable。嗯,这样,我们可以试试JDK1.5的新特性:可变参数。
1 /** 2 * 封装的关闭方法 3 * @param ios:要关闭的流 4 */ 5 public static void close(Closeable... ios) { 6 for(Closeable io:ios) { 7 try { 8 if(io!=null) { 9 io.close(); 10 } 11 }catch(IOException e) { 12 e.printStackTrace(); 13 } 14 } 15 }
现在看似完美了,但我还不太满意,有时候,我觉得手动关闭资源太麻烦了。别急,来看看JDK1.7的新特性:try...with...resources(自动关闭资源)。
1 /** 2 * JDK1.7之后的新特性 try...with...resources:自动关闭资源 3 * 文件的拷贝 4 * @param srcPath:要拷贝的源头 5 * @param destPath:要拷贝到的目的地 6 */ 7 public static void copy1(String srcPath,String destPath) { 8 //1.创建源 9 File src = new File(srcPath); 10 File dest = new File(destPath); 11 //2.选择流 12 try(InputStream is = new FileInputStream(src); 13 OutputStream os = new FileOutputStream(dest)) { 14 //3.操作 15 byte[] flush = new byte[1024]; //缓冲容器 16 int len; //接收长度 17 while((len=is.read(flush))!=-1) { 18 os.write(flush, 0, len); 19 } 20 os.flush(); 21 }catch(FileNotFoundException e) { 22 e.printStackTrace(); 23 }catch(IOException e) { 24 e.printStackTrace(); 25 } 26 }
大家发现:try里面写的好繁琐,别急,看看JDK1.9的改进版(不过要求你所要关闭的资源是final或等效于final的变量)。
1 /** 2 * JDK1.9之后对 try...with...resources的改进 3 * 对接流 4 * @param is:输入流 5 * @param os:输出流 6 */ 7 public static void copy2(InputStream is,OutputStream os) { 8 //1.创建源 9 //2.选择流 10 try(is;os) { 11 //3.操作 12 byte[] flush = new byte[1024]; //缓冲容器 13 int len; //接收长度 14 while((len=is.read(flush))!=-1) { 15 os.write(flush, 0, len); 16 } 17 os.flush(); 18 }catch(FileNotFoundException e) { 19 e.printStackTrace(); 20 }catch(IOException e) { 21 e.printStackTrace(); 22 } 23 }
哈哈,看上去是不是简洁很多。对的,随着JDK的新版本发布,越来越多的新技术,也使得代码看起来越简洁,不过对我们的要求也只会越来越高。前段时间JDK12已经出来了,还没用,不过我相信肯定会有好多的新特性,期待,也看好Java,加油。