JavaIO流中的拷贝
JavaIO流中对数据的操作尤为重要,掌握了基本的拷贝操作,才能将各种数据源的操作联系起来。
先来看看对文件夹的拷贝吧:
/**
* 利用递归实现文件夹的拷贝操作
* 分析:判断
* 是文件:调用拷贝文件的方法fileCopy(...)
* 是文件夹:创建文件夹,并使用递归实现子文件夹/子文件的判断及操作
* @param src:要拷贝的文件夹源头
* @param dest:要拷贝到的文件夹源头
*/
public static void dirCopy(File src,File dest) {
if(src.isFile()) { //是文件
fileCopy(src, dest);
}else { //是文件夹
dest.mkdirs();
for(File subSrc:src.listFiles()) { //遍历子文件夹/子文件
dirCopy(subSrc, new File(dest,subSrc.getName()));
}
}
}
对文件的拷贝,我们可以这样写:
/**
* 实现文件的拷贝
* 输入流读取的同时输出流进行写出
* @param src:要拷贝的文件源头
* @param dest:要拷贝到的文件源头
*/
public static void fileCopy(File src,File dest) {
//1.创建源
//2.选择流
InputStream is = null;
OutputStream os = null;
try {
is = new FileInputStream(src);
os = new FileOutputStream(dest);
//3.操作:分段读取并写出
int len; //接收长度
byte[] flush = new byte[1024]; //缓冲容器,一次读写1k
while((len=is.read(flush))!=-1) {
os.write(flush, 0, len);
}
os.flush(); //写完手动刷新,避免数据在缓冲容器中(当然当流关闭时会自动刷新)
}catch(FileNotFoundException e) {
e.printStackTrace();
}catch(IOException e) {
e.printStackTrace();
}finally {
//4.关闭流,分别关闭,先打开的后关闭
try {
if(os!=null) { //判断是否为空,避免空指针异常
os.close();
}
}catch(IOException e) {
e.printStackTrace();
}
try {
if(is!=null) { //判断是否为空,避免空指针异常
is.close();
}
}catch(IOException e) {
e.printStackTrace();
}
}
}
以上代码只能实现对文件的拷贝操作,当然适合于拷贝任何格式的数据文件,包括视频、音频、图片等等。但是如果我想将一张图片拷贝到字节数组中呢(这里的字节数组相当于内存),也就是说从文件到字节数组,或着是从字节数组到文件。那么以上代码就具有局限性了,也不易于扩展,来看以下代码:
/**
* 对接流
* @param is:输入流
* @param os:输出流
*/
public static void copy(InputStream is,OutputStream os) {
//1.创建源
//2.选择流
try {
//3.操作
byte[] flush = new byte[1024]; //缓冲容器
int len; //接收长度
while((len=is.read(flush))!=-1) {
os.write(flush, 0, len);
}
os.flush();
}catch(FileNotFoundException e) {
e.printStackTrace();
}catch(IOException e) {
e.printStackTrace();
}finally {
try {
if(os!=null) {
os.close();
}
}catch(IOException e) {
e.printStackTrace();
}
try {
if(is!=null) {
is.close();
}
}catch(IOException e) {
e.printStackTrace();
}
}
}
嗯,这样就可以实现以上要求了,但是我们发现:关闭资源的操作一直在,而且都一样,我们可以封装一下,这样在finally中就可以直接调用了。
/**
* 关闭的方法
* @param is:输入流
* @param os:输出流
*/
public static void close(InputStream is,OutputStream os) {
try {
if(os!=null) {
os.close();
}
}catch(IOException e) {
e.printStackTrace();
}
try {
if(is!=null) {
is.close();
}
}catch(IOException e) {
e.printStackTrace();
}
}
好了,看看封装的代码,幸亏只有两个流,要是流很多咋办,形参太多,但是我们发现输入流InputStream和输出流OutputStream都实现了同一个接口:Closeable。嗯,这样,我们可以试试JDK1.5的新特性:可变参数。
/**
* 封装的关闭方法
* @param ios:要关闭的流
*/
public static void close(Closeable... ios) {
for(Closeable io:ios) {
try {
if(io!=null) {
io.close();
}
}catch(IOException e) {
e.printStackTrace();
}
}
}
现在看似完美了,但我还不太满意,有时候,我觉得手动关闭资源太麻烦了。别急,来看看JDK1.7的新特性:try...with...resources(自动关闭资源)。
/**
* JDK1.7之后的新特性 try...with...resources:自动关闭资源
* 文件的拷贝
* @param srcPath:要拷贝的源头
* @param destPath:要拷贝到的目的地
*/
public static void copy1(String srcPath,String destPath) {
//1.创建源
File src = new File(srcPath);
File dest = new File(destPath);
//2.选择流
try(InputStream is = new FileInputStream(src);
OutputStream os = new FileOutputStream(dest)) {
//3.操作
byte[] flush = new byte[1024]; //缓冲容器
int len; //接收长度
while((len=is.read(flush))!=-1) {
os.write(flush, 0, len);
}
os.flush();
}catch(FileNotFoundException e) {
e.printStackTrace();
}catch(IOException e) {
e.printStackTrace();
}
}
大家发现:try里面写的好繁琐,别急,看看JDK1.9的改进版(不过要求你所要关闭的资源是final或等效于final的变量)。
/**
* JDK1.9之后对 try...with...resources的改进
* 对接流
* @param is:输入流
* @param os:输出流
*/
public static void copy2(InputStream is,OutputStream os) {
//1.创建源
//2.选择流
try(is;os) {
//3.操作
byte[] flush = new byte[1024]; //缓冲容器
int len; //接收长度
while((len=is.read(flush))!=-1) {
os.write(flush, 0, len);
}
os.flush();
}catch(FileNotFoundException e) {
e.printStackTrace();
}catch(IOException e) {
e.printStackTrace();
}
}
哈哈,看上去是不是简洁很多。对的,随着JDK的新版本发布,越来越多的新技术,也使得代码看起来越简洁,不过对我们的要求也只会越来越高。前段时间JDK12已经出来了,还没用,不过我相信肯定会有好多的新特性,期待,也看好Java,加油。
JavaIO流中的拷贝的更多相关文章
- 总结JAVA----IO流中的字节流
对于IO流中字节流的总结 字节流的概念 由于应用程序,经常需要和文件打交道,所以Inputstream专门提供了读写文件的子类:FileInputStream和FileOutputStream类,如果 ...
- 总结JAVA----IO流中的File类
对于IO流中File类的总结 File类的基本概念 File类只能用于完成对于文件属性(是否存在.可读性.长度)的一些操作,不能用于文件的访问. File类的对象 File类的对象存储的是文件的绝对路 ...
- javaIO流实现文件拷贝
package com.java.demo; import java.io.*; public class CopyDemo { public static void main(String[] ar ...
- JAVA-IO流大文件拷贝
package com.test.io; import java.io.BufferedInputStream; import java.io.BufferedOutputStream; import ...
- java-IO流-字符流-FileReader、FileWriter、自定义小数组的拷贝、BufferedReader、BufferedWriter、readLine()和newLine()方法、LineNumberReader、使用指定的码表读写字符
###21.01_IO流(字符流FileReader) * 1.字符流是什么 * 字符流是可以直接读写字符的IO流 * 字符流读取字符, 就要先读取到字节数据, 然后转为字符. 如果要 ...
- JavaIO流(输入输出操作)
Java中执行输出和输入操作,需要通过IO流.例如最常见的System.out.println()就是一个输出流.IO流的类比较多,但核心体系就是由File. InputStream .OutputS ...
- Java学习日记之 Java-IO流
Java中的IO流在处理上分为字节流和字符流.字节流和字符流的区别 : 1.字节流读取的时候,读到一个字节就返回一个字节: 字符流使用了字节流读到一个或多个字节(中文对应的字节数是两个,在UTF-8 ...
- .net学习之集合、foreach原理、Hashtable、Path类、File类、Directory类、文件流FileStream类、压缩流GZipStream、拷贝大文件、序列化和反序列化
1.集合(1)ArrayList内部存储数据的是一个object数组,创建这个类的对象的时候,这个对象里的数组的长度为0(2)调用Add方法加元素的时候,如果第一次增加元神,就会将数组的长度变为4往里 ...
- java中的拷贝(二)深克隆
浅拷贝(Object类中的clone()方法)是指在拷贝对象时,对于基本数据类型的变量会重新复制一份,而对于引用类型的变量只是对引用进行拷贝. 深拷贝(或叫深克隆) 则是对对象及该对象关联的对象内容, ...
随机推荐
- Confluence 6 编辑站点欢迎消息
通过编辑欢迎信息能够为你站点的主页面添加一些个人信息. 站点的欢迎信息显示在站点主面板的右侧,这是你对站点添加声明,连接,有关你项目组美好回忆照片的完美位置. 你需要 Confluence 管理员权限 ...
- Python基础之递归函数与二分法
一.递归函数 定义: 在函数内部,可以调用其他函数.如果一个函数在内部调用自身本身,这个函数就是递归函数. 我们来举个例子吧,比如:有个人问“egon”年龄,他说比“小大”大5岁,“小大”又说比“小保 ...
- CF 494B 【Obsessive String】
很有趣的一道题 这道题提议很难懂,其实就是让你求合法的集合数目.合法的集合定义为: 1.集合中的所有串都是s的子串,且互不重叠 2.集合中的所有串都含有子串t. 看到网上很多题解说要用kmp,但我就不 ...
- 简单的做一个图片上传预览(web前端)
转载:点击查看原文 在做web项目很多的时候图片都是避免不了的,所以操作图片就成了一个相对比较棘手的问题,其实也不是说很麻烦,只是说上传然后直接预览的过程很恶心,今天简单的做一个处理. 效果预览: & ...
- mysql 去除重复 Select中DISTINCT关键字的用法 在使用mysql时,有时需要查询出某个字段不重复的记录,虽然mysql提供 有distinct这个关键字来过滤掉多余的重复记录只保留一条,但往往只用它来返回不重复记录的条数,而不是用它来返回不重记录的所有值。其原因是 distinct只能返回它的目标字段,而无法返回其它字段,这个问题让我困扰了很久,用distinct不能解决的话,
在使用mysql时,有时需要查询出某个字段不重复的记录,虽然mysql提供 有distinct这个关键字来过滤掉多余的重复记录只保留一条,但往往只用它来返回不重复记录的条数,而不是用它来返回不重记 ...
- C++ Primer 笔记——关联容器
1.关联容器支持高效的关键字查找和访问,标准库提供8个关联容器. 2.如果一个类型定义了“行为正常”的 < 运算符,则它可以用作关键字类型. 3.为了使用自己定义的类型,在定义multiset时 ...
- 论文阅读笔记二十二:End-to-End Instance Segmentation with Recurrent Attention(CVPR2017)
论文源址:https://arxiv.org/abs/1605.09410 tensorflow 代码:https://github.com/renmengye/rec-attend-public 摘 ...
- 如何使用Scrapy框架实现网络爬虫
现在用下面这个案例来演示如果爬取安居客上面深圳的租房信息,我们采取这样策略,首先爬取所有租房信息的链接地址,然后再根据爬取的地址获取我们所需要的页面信息.访问次数多了,会被重定向到输入验证码页面,这个 ...
- 史上最简单的 SpringCloud 教程
史上最简单的 SpringCloud 教程 | 第一篇: 服务的注册与发现(Eureka)史上最简单的SpringCloud教程 | 第二篇: 服务消费者(rest+ribbon)史上最简单的Spri ...
- C/C++内存管理器
C标准库提供了malloc,free,calloc,realloc,C++标准库还提供了new, new[], delete, delete[].这些用来管理内存,看起来够用了,为啥还要自己写一个内存 ...