FileStream大文件复制

      FileStream缓冲读取和写入可以提高性能。FileStream读取文件的时候,是先讲流放入内存,经Flash()方法后将内存中(缓冲中)的数据写入文件。如果文件非常大,势必消耗性能。特封装在FileHelper中以备不时之需。参考文章:http://www.cnblogs.com/yangxiaohu1/archive/2008/06/20/1226949.html将该文章中提供的代码少做修改,原文中进行了强制类型转换,如果文件很大,比如4G,就会出现溢出的情况,复制的结果字节丢失严重,导致复制文件和源文件大小不一样。这里修改的代码如下:

 1  public static class FileHelper
 2     {
 3         /// <summary>
 4         /// 复制大文件
 5         /// </summary>
 6         /// <param name="fromPath">源文件的路径</param>
 7         /// <param name="toPath">文件保存的路径</param>
 8         /// <param name="eachReadLength">每次读取的长度</param>
 9         /// <returns>是否复制成功</returns>
10         public static bool CopyFile(string fromPath, string toPath, int eachReadLength)
11         {
12             //将源文件 读取成文件流
13             FileStream fromFile = new FileStream(fromPath, FileMode.Open, FileAccess.Read);
14             //已追加的方式 写入文件流
15             FileStream toFile = new FileStream(toPath, FileMode.Append, FileAccess.Write);
16             //实际读取的文件长度
17             int toCopyLength = 0;
18             //如果每次读取的长度小于 源文件的长度 分段读取
19             if (eachReadLength < fromFile.Length)
20             {
21                 byte[] buffer = new byte[eachReadLength];
22                 long copied = 0;
23                 while (copied <= fromFile.Length - eachReadLength)
24                 {
25                     toCopyLength = fromFile.Read(buffer, 0, eachReadLength);
26                     fromFile.Flush();
27                     toFile.Write(buffer, 0, eachReadLength);
28                     toFile.Flush();
29                     //流的当前位置
30                 toFile.Position = fromFile.Position;
31                     copied += toCopyLength;
32                     
33                 }
34                 int left = (int)(fromFile.Length - copied);
35                 toCopyLength = fromFile.Read(buffer, 0, left);
36                 fromFile.Flush();
37                 toFile.Write(buffer, 0, left);
38                 toFile.Flush();
39 
40             }
41             else
42             {
43                 //如果每次拷贝的文件长度大于源文件的长度 则将实际文件长度直接拷贝
44              byte[] buffer = new byte[fromFile.Length];
45                 fromFile.Read(buffer, 0, buffer.Length);
46                 fromFile.Flush();
47                 toFile.Write(buffer, 0, buffer.Length);
48                 toFile.Flush();
49 
50             }
51             fromFile.Close();
52             toFile.Close();
53             return true;
54         }
55     }

测试代码:

 1  class Program
 2     {
 3         static void Main(string[] args)
 4         {
 5 
 6             Stopwatch watch = new Stopwatch();
 7             watch.Start();
 8             if (FileHelper.CopyFile(@"D:\安装文件\新建文件夹\SQLSVRENT_2008R2_CHS.iso", @"F:\SQLSVRENT_2008R2_CHS.iso", 1024 * 1024 * 5))
 9             {
10                 watch.Stop();
11                 Console.WriteLine("拷贝完成,耗时:" + watch.Elapsed.Seconds+"秒");
12 
13             }
14             Console.Read();
15         }
16 
17     }

结果:

MD5校验结果:

文件: D:\安装文件\新建文件夹\SQLSVRENT_2008R2_CHS.iso
大小: 4662884352 字节
修改时间: 2010年9月3日, 10:41:26
MD5: D2BC1D35D987CC6CB8401BFB0A1E1BC9
SHA1: 0EEFF017B21635DF33F33C47E31E911CB23390F7
CRC32: 55AC3C56

文件: F:\SQLSVRENT_2008R2_CHS.iso
大小: 4662884352 字节
修改时间: 2013年9月29日, 10:51:39
MD5: D2BC1D35D987CC6CB8401BFB0A1E1BC9
SHA1: 0EEFF017B21635DF33F33C47E31E911CB23390F7
CRC32: 55AC3C56

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏张俊红

python数据分析笔记——数据加载与整理

Python数据分析——数据加载与整理 总第47篇 ▼ ? (本文框架) 数据加载 导入文本数据 ? 1、导入文本格式数据(CSV)的方法: 方法一:使用pd....

42280
来自专栏岑玉海

hbase源码系列(十三)缓存机制MemStore与Block Cache

这一章讲hbase的缓存机制,这里面涉及的内容也是比较多,呵呵,我理解中的缓存是保存在内存中的特定的便于检索的数据结构就是缓存。 之前在讲put的时候,put是...

54370
来自专栏Elasticsearch实验室

Elasitcsearch 底层系列 Lucene 内核解析之 Stored Fields

Lucene 的 stored fields 主要用于行存文档需要保存的字段内容,每个文档的所有 stored fields 保存在一起,在查询请求需要返回字段...

90450
来自专栏分布式系统进阶

Kafka消息的磁盘存储Kafka源码分析-汇总

可以看到使用FileMessageSet来操作Log文件, 使用OffsetIndex来操作Index文件

37120
来自专栏小灰灰

Java 动手写爬虫: 三、爬取队列

第三篇 爬取队列的实现 第二篇中,实现了深度爬取的过程,但其中一个比较明显的问题就是没有实现每个爬取作为一个独立的任务来执行;即串行的爬取网页中的链接;因此,...

56050
来自专栏逆向技术

16位汇编语言第二讲系统调用原理,以及各个寄存器详解

   16位汇编语言第二讲系统调用原理,以及各个寄存器详解 昨天已将简单的写了一下汇编代码,并且执行了第一个显示到屏幕的helloworld 问题?   hel...

25600
来自专栏分布式系统和大数据处理

C#网络编程(同步传输字符串) - Part.2

在与服务端的连接建立以后,我们就可以通过此连接来发送和接收数据。端口与端口之间以流(Stream)的形式传输数据,因为几乎任何对象都可以保存到流中,所以实际上可...

14130
来自专栏JadePeng的技术博客

Angular快速学习笔记(4) -- Observable与RxJS

98220
来自专栏DOTNET

ASP.NET MVC编程——控制器

每一个请求都会经过控制器处理,控制器中的每个方法被称为控制器操作,它处理具体的请求。 1操作输入参数 控制器的操作的输入参数可以是内置类型也可以是自定义类型。 ...

30390
来自专栏杨建荣的学习笔记

两个关于权限设置的问题思考

最近这两天做动态菜单和权限校验,想到了两个有意思的问题。 第一个是对于一个用户的操作权限,无非就是这四个方面,增删改查。 如果通过字母来标识,可能就是增(I)删...

36170

扫码关注云+社区

领取腾讯云代金券