Java IO与NIO来Copy文件的四种方法实现以及性能对比

使用Java的IO与NIO来Copy文件的四种方法实现以及性能对比

FileCopyRunner接口,定义了Copy文件的接口,等下在测试类中使用匿名内部类来实现。

package nio.channel;import java.io.File;public interface FileCopyRunner {void copyFile(File source , File target);
}

测试类:

  • benchmark():Copy文件ROUNDS(5次,并且返回耗费的平均时间(1.0F)*elapsed / ROUNDS
  • close():关闭资源。

完整代码(下面说四种方法):

package nio.channel;import java.io.*;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;public class FileCopyDemo {private static final int ROUNDS = 5;private static void benchmark(FileCopyRunner test ,File sourse , File target){long elapsed = 0L;for (int i = 0; i < ROUNDS; i++) {long startTime = System.currentTimeMillis();test.copyFile(sourse , target);elapsed += System.currentTimeMillis() - startTime;target.delete();}System.out.println(test+":"+(1.0F)*elapsed / ROUNDS);}public static void close(Closeable closeable){if(closeable != null){try {closeable.close();} catch (IOException e) {e.printStackTrace();}}}public static void main(String[] args) {FileCopyRunner noBufferStreamCopy  = new FileCopyRunner() {@Overridepublic void copyFile(File sourse, File target) {InputStream fin = null;OutputStream fout = null;try {fin = new FileInputStream(sourse);fout = new FileOutputStream(target);int result;while((result = fin.read()) != -1){fout.write(result);}} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} finally{close(fin);close(fout);}}@Overridepublic String toString() {return "noBufferStreamCopy";}};FileCopyRunner bufferedStreamCopy = new FileCopyRunner() {@Overridepublic void copyFile(File sourse, File target) {InputStream fin = null;OutputStream fout = null;try {fin = new BufferedInputStream(new FileInputStream(sourse));fout = new BufferedOutputStream(new FileOutputStream(target));byte[] buffer = new byte[8192];int result;while((result = fin.read(buffer)) != -1){fout.write(buffer , 0 ,result);}} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} finally{close(fin);close(fout);}}@Overridepublic String toString() {return "bufferedStreamCopy";}};FileCopyRunner nioBufferCopy = new FileCopyRunner() {@Overridepublic void copyFile(File sourse, File target) {FileChannel fin = null;FileChannel fout = null;try {fin = new FileInputStream(sourse).getChannel();fout = new FileOutputStream(target).getChannel();ByteBuffer buffer = ByteBuffer.allocate(8192);while(fin.read(buffer) != -1){buffer.flip(); //开始读模式while(buffer.hasRemaining()){fout.write(buffer);}buffer.clear(); // 开始写模式}} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} finally{close(fin);close(fout);}}@Overridepublic String toString() {return "nioBufferCopy";}};FileCopyRunner nioTransferCopy = new FileCopyRunner() {@Overridepublic void copyFile(File sourse, File target) {FileChannel fin = null;FileChannel fout = null;try {fin = new FileInputStream(sourse).getChannel();fout = new FileOutputStream(target).getChannel();long transferred = 0;long size = fin.size();while(transferred != size){transferred += fin.transferTo(0,size,fout);}} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} finally{close(fin);close(fout);}}@Overridepublic String toString() {return "nioTransferCopy";}};File one = new File("E:\\test\\1.png");File oneCopy = new File("E:\\test\\1-copy.png");System.out.println("---Copying one---");benchmark(noBufferStreamCopy , one , oneCopy);benchmark(bufferedStreamCopy , one , oneCopy);benchmark(nioBufferCopy , one , oneCopy);benchmark(nioTransferCopy , one , oneCopy);File two = new File("E:\\test\\2.mp4");File twoCopy = new File("E:\\test\\2-copy.mp4");System.out.println("---Copying two---");
//        benchmark(noBufferStreamCopy , two , twoCopy);benchmark(bufferedStreamCopy , two , twoCopy);benchmark(nioBufferCopy , two , twoCopy);benchmark(nioTransferCopy , two , twoCopy);File three = new File("E:\\test\\3.mp4");File threeCopy = new File("E:\\test\\3-copy.mp4");System.out.println("---Copying three---");
//        benchmark(noBufferStreamCopy , three , threeCopy);benchmark(bufferedStreamCopy , three , threeCopy);benchmark(nioBufferCopy , three , threeCopy);benchmark(nioTransferCopy , three , threeCopy);File four = new File("E:\\test\\4.avi");File fourCopy = new File("E:\\test\\4-copy.avi");System.out.println("---Copying four---");
//        benchmark(noBufferStreamCopy , four , fourCopy);benchmark(bufferedStreamCopy , four , fourCopy);benchmark(nioBufferCopy , four , fourCopy);benchmark(nioTransferCopy , four , fourCopy);}
}

匿名内部类一:

使用FileInputStreamFileOutputStream来Copy文件,它是一个字节一个字节进行read的,所以也是一个字节一个字节进行write的,read()的源码注释很清楚的写出来了,所以这种方法的性能特别差,等下用较大文件测试时,我们选择跳过这种方法(因为太久了),内部逻辑应该很简单吧,从read()的源码注释可以知道read()的返回值是介于0-255的值,其实就是读取的一个字节(8位)The value byte is returned as an int in the range 0 to 255。

     * Reads the next byte of data from the input stream. The value byte is* returned as an <code>int</code> in the range <code>0</code> to* <code>255</code>. If no byte is available because the end of the stream* has been reached, the value <code>-1</code> is returned. This method* blocks until input data is available, the end of the stream is detected,* or an exception is thrown.
        FileCopyRunner noBufferStreamCopy  = new FileCopyRunner() {@Overridepublic void copyFile(File source, File target) {InputStream fin = null;OutputStream fout = null;try {fin = new FileInputStream(source);fout = new FileOutputStream(target);int result;while((result = fin.read()) != -1){fout.write(result);}} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} finally{close(fin);close(fout);}}@Overridepublic String toString() {return "noBufferStreamCopy";}};

匿名内部类二:

第二种方法使用BufferedInputStreamBufferedOutputStream来进行文件的Copy,它们会产生一个缓冲区,默认大小都是8192字节,源码如下:

    private static int DEFAULT_BUFFER_SIZE = 8192;/*** Creates a <code>BufferedInputStream</code>* and saves its  argument, the input stream* <code>in</code>, for later use. An internal* buffer array is created and  stored in <code>buf</code>.** @param   in   the underlying input stream.*/public BufferedInputStream(InputStream in) {this(in, DEFAULT_BUFFER_SIZE);}
    /*** Creates a new buffered output stream to write data to the* specified underlying output stream.** @param   out   the underlying output stream.*/public BufferedOutputStream(OutputStream out) {this(out, 8192);}

利用缓冲区,会大大提升性能,因为避免了频繁的打开、关闭文件,有了缓冲区,我们每次对文件进行读、写操作,都可以读、写更多字节数据,减少了打开、关闭文件等操作的次数。

        FileCopyRunner bufferedStreamCopy = new FileCopyRunner() {@Overridepublic void copyFile(File source, File target) {InputStream fin = null;OutputStream fout = null;try {fin = new BufferedInputStream(new FileInputStream(source));fout = new BufferedOutputStream(new FileOutputStream(target));byte[] buffer = new byte[8192];int result;while((result = fin.read(buffer)) != -1){fout.write(buffer , 0 ,result);}} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} finally{close(fin);close(fout);}}@Overridepublic String toString() {return "bufferedStreamCopy";}};

 

匿名内部类三:

第三种方法使用NIO中的ChannelBuffer来进行Copy文件。

Java 网络编程-NIO原理概述

和上一种方法一样,这里创建8192字节Buffer,方便进行性能的对比。

ByteBuffer buffer = ByteBuffer.allocate(8192);

下面两种操作,大家应该知道吧,不知道就往下看。

buffer.flip(); //开始读模式
buffer.clear(); // 开始写模式

 源码如下(先不管mark有什么用):

    private int position = 0;private int limit;private int capacity;public final Buffer flip() {limit = position;position = 0;mark = -1;return this;}public final Buffer clear() {position = 0;limit = capacity;mark = -1;return this;}

 

下面这张图应该描述的很清楚。

写模式:position位置之前的数据都是写入的(包括position位置),每写入一字节数据,position++,所以clear()position置0,是不是就说明开始准备要写入了,并且limit = capacity,即进入写模式。
读模式:写入数据后,0位置到position位置的数据都是写入的,调用flip(),将limit置成position,而position置成0,所以position(0)到limit(原position位置)是不是就是要被读取的数据范围,即进入读模式


 

        FileCopyRunner nioBufferCopy = new FileCopyRunner() {@Overridepublic void copyFile(File source, File target) {FileChannel fin = null;FileChannel fout = null;try {fin = new FileInputStream(source).getChannel();fout = new FileOutputStream(target).getChannel();ByteBuffer buffer = ByteBuffer.allocate(8192);while(fin.read(buffer) != -1){buffer.flip(); //开始读模式while(buffer.hasRemaining()){fout.write(buffer);}buffer.clear(); // 开始写模式}} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} finally{close(fin);close(fout);}}@Overridepublic String toString() {return "nioBufferCopy";}};

匿名内部类四:

第四种方法只使用NIOChannel来进行文件的Copy,Channel通过transferTo()可以把数据写入另一个Channel
transferTo()的源码注释也可以看出。

Transfers bytes from this channel’s file to the given writable byte channel.

 

        FileCopyRunner nioTransferCopy = new FileCopyRunner() {@Overridepublic void copyFile(File source, File target) {FileChannel fin = null;FileChannel fout = null;try {fin = new FileInputStream(source).getChannel();fout = new FileOutputStream(target).getChannel();long transferred = 0;long size = fin.size();while(transferred != size){transferred += fin.transferTo(0,size,fout);}} catch (FileNotFoundException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();} finally{close(fin);close(fout);}}@Overridepublic String toString() {return "nioTransferCopy";}};

 

性能对比

文件详情如下:

  • 1.png(37KB)。
  • 2.mp4(3.92MB)。
  • 3.mp4(132MB)。
  • 4.avi(638MB)。

每种方法Copy文件5次,计算平均耗时(第一种方法除外,因为它太慢了)。

输出:

---Copying one---
noBufferStreamCopy:143.4
bufferedStreamCopy:0.4
nioBufferCopy:1.0
nioTransferCopy:0.4
---Copying two---
bufferedStreamCopy:5.0
nioBufferCopy:5.8
nioTransferCopy:2.6
---Copying three---
bufferedStreamCopy:161.0
nioBufferCopy:154.0
nioTransferCopy:90.8
---Copying four---
bufferedStreamCopy:1659.0
nioBufferCopy:1294.4
nioTransferCopy:1266.6

我测试了很多次,需要缓冲区的方法,性能跟文件大小、缓冲区大小都有关系。

不过,第四种方法性能还是挺不错的。

如果有说错的地方,请大家不吝赐教

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/14052.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

广告投放—常见术语

01 按...计费 英文中文场景CPT按时间计费品牌采买CPM每千人展现成本CPM总消费/曝光量*1000CPC按点击收费竞价CPD按下载收费 CPS 按销售收费佣金&#xff0c;如&#xff1a;销售额1000&#xff0c;CPS3%&#xff0c;广告费30CPA按行为收费&#xff08;行为&#xff1a;下载、注…

【C语言】明析部分C语言内存函数

目录 1.memcpy 2.memmove 3.memset 4.memcmp 以下都是内存函数&#xff0c;作用单位均是字节 1.memcpy memcpy是C/C语言中的一个内存拷贝函数&#xff0c;其原型为&#xff1a; void* memcpy(void* dest, const void* src, size_t n);目标空间&#xff08;字节&#xff09…

C语言习题~day17

1.下面代码关于数组名描述不正确的是&#xff08; &#xff09; int main() {int arr[10] {0};return 0; } A.数组名arr和&arr是一样的 B.sizeof(arr)&#xff0c;arr表示整个数组 C.&arr&#xff0c;arr表示整个数组 D.除了sizeof(arr)和&arr中的数组名&…

三生随记——面试之夜

深夜&#xff0c;林浩独自一人站在一座孤零零的大楼前。这座大楼隐藏在城市的边缘&#xff0c;四周弥漫着浓重的雾气&#xff0c;仿佛吞噬着一切光明。他紧紧握住手中的简历&#xff0c;那是他唯一的希望&#xff0c;也是唯一能与外界联系的东西。 他在网上看到一份诱人的工作广…

【Linux系统化学习】应用层——HTTPS协议

目录 什么是加密、解密 为什么要加密 臭名昭著的”运营商劫持“ 常见的加密方式 对称加密 非对称加密 数据摘要&&数据指纹 两个用途 HTTPS的工作过程探究 方案1——只是用对称加密 方案2——只使用非对称加密 方案3——双方都是用非对称加密 方案4——对称…

#05【面试问题整理】嵌入式软件工程师

前言 本系列博客主要记录有关嵌入式方面的面试重点知识,本系列已经更新的篇目有如下: ​ 1.1进程线程的基本概念 1.2 并发,同步,异步,互斥,阻塞,非阻塞的理解 1.3 孤儿进程、僵尸进程、守护进程的概念 【本篇】5.1 Linux内核相关 6.0 单片机常见面试题 内容如有错误请在…

科技引领乡村振兴新潮流:运用现代信息技术手段,提升农业生产和乡村管理效率,打造智慧化、现代化的美丽乡村

一、引言 随着科技的不断进步&#xff0c;现代信息技术已经渗透到社会的各个领域&#xff0c;成为推动社会发展的重要力量。在乡村振兴战略的背景下&#xff0c;科技的力量同样不容忽视。本文旨在探讨如何运用现代信息技术手段&#xff0c;提升农业生产和乡村管理效率&#xf…

js如何遍历FormData的值

遍历FormData的值&#xff0c;一般有2种方法&#xff1a;forEach 和 for...of entries const data new FormData();data.append(aaa, 111); data.append(bbb, 222);// 方法1 data.forEach((value, key) > {console.log(key, value); }) 输出 aaa 111 和 bbb 222// 方法2 …

java自学阶段二:JavaWeb开发--day04(Maven学习)

day04学习笔记 一、学习目标 1.了解maven的基础概念&#xff1b; 2.学会maven的部署&#xff1b; 3.maven的作用&#xff1a;标准化&#xff1b;方便找依赖 maven就是一个开源项目&#xff0c;专门用来管理和构建Java项目的&#xff1b;我们自己写的项目结构可能会千奇百怪&am…

每日一题《leetcode--116.填充每个结点的下一个右侧结点》

https://leetcode.cn/problems/populating-next-right-pointers-in-each-node/ 题目要求给每个结点的next指针进行填充&#xff0c;使每个结点的next指针指向其下一个右侧结点。如果右侧没有结点&#xff0c;则将next指针设置为空。 struct Node* connect(struct Node* root) {…

快速搭建 WordPress 外贸电商网站指南

本指南全面解析了在 Hostinger 平台上部署 WordPress 外贸电商网站的详细步骤&#xff0c;涵盖托管方案选择、WordPress 一键安装、主题挑选与演示数据导入、主题个性化定制、SEO插件插件 AIOSEO 安装、通过 GTranslate 实现多语言自动翻译、地区访问控制插件&#xff0c;助力用…

人工智能(Educoder)-- 机器学习 -- 神经网络(初级)

第一关 注&#xff1a; 神经网络的起源和应用 起源&#xff1a;神经网络最早由心理学家和神经学家开创&#xff0c;目的是模拟生物神经系统对真实世界物体的交互反应。应用&#xff1a;现代神经网络用于分类&#xff08;如图像识别、文本分类&#xff09;和数值预测&#xff08…

567. 字符串的排列(c++)滑动窗口

给你两个字符串 s1 和 s2 &#xff0c;写一个函数来判断 s2 是否包含 s1 的排列。如果是&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 换句话说&#xff0c;s1 的排列之一是 s2 的 子串 。 示例 1&#xff1a; 输入&#xff1a;s1 "ab" s2 …

OSI参考模型中数据的封装和解封过程

OSI&#xff08;开放系统互联&#xff09;参考模型是一种网络协议分层架构模型&#xff0c;它将网络通信过程划分为七个层次。数据在每一层都要进行相应的封装处理&#xff0c;具体过程如下&#xff1a; 1. 应用层&#xff08;Application Layer&#xff09; 功能&#xff1a…

最小堆的数组实现

堆是一棵完全二叉树&#xff0c;之所以需要堆&#xff0c;是因为我们需要堆序性&#xff0c;堆的父节点都大于或小于其子节点&#xff0c;这样的有序性能让我们快速找到最大值或最小值&#xff0c;即根节点&#xff0c;时间复杂度是O&#xff08;1&#xff09; 由于完全二叉树…

TS tsconfig.json配置项

files - 设置要编译的文件的名称&#xff1b;include - 设置需要进行编译的文件&#xff0c;支持路径模式匹配&#xff1b;exclude - 设置无需进行编译的文件&#xff0c;支持路径模式匹配&#xff1b;compilerOptions - 设置与编译流程相关的选项。 compilerOptions&#xff…

Windows下安装部署rocketmq

1.1.下载安装rocketmq 下载 | RocketMQ 下载完后解压到自定义目录&#xff0c;MQ解压路径\rocketmq-all-4.6.0-bin-release&#xff1b;&#xff08;Windows10系统解压路径不要出现空格&#xff09; 1.2.配置环境变量 配置环境变量&#xff0c;变量名&#xff1a;ROCKETM…

python输出水仙花数两种方法

在Python中&#xff0c;可以使用数学优化方法来输出所有的水仙花数。水仙花数是指一个三位数&#xff0c;其各位数字的立方和等于该数本身。例如&#xff0c;153是一个水仙花数&#xff0c;因为1^3 5^3 3^3 153。 下面我们介绍两种主要方法来输出所有的水仙花数。 方法一&…

深入探索C/C++内存管理

目录 C/C内存分布 C语言中动态内存管理方式 calloc realloc free C中动态内存管理方式 new和delete操作内置类型 new和delete操作自定义类型 operator new和operator delete函数 new和delete的实现原理 内置类型 自定义类型 定位new和表达式(placement-new) 常见面试…

C++进阶之路:何为运算符重载、赋值运算符重载与前后置++重载(类与对象_中篇)

✨✨ 欢迎大家来访Srlua的博文&#xff08;づ&#xffe3;3&#xffe3;&#xff09;づ╭❤&#xff5e;✨✨ &#x1f31f;&#x1f31f; 欢迎各位亲爱的读者&#xff0c;感谢你们抽出宝贵的时间来阅读我的文章。 我是Srlua小谢&#xff0c;在这里我会分享我的知识和经验。&am…