如何在没有 CopyFile 或 CopyFileEx 的情况下在 Windows 上复制大文件?
题
Windows Server 2003 存在一个限制,阻止您复制极大的文件(与您拥有的 RAM 量成比例)。限制在于 CopyFile 和 CopyFileEx 函数,它们由 xcopy、Explorer、Robocopy 和 .NET FileInfo 类使用。
这是您收到的错误:
无法复制[文件名]:系统资源不足,无法完成请求的服务。
这是一个 知识库文章 关于这个主题,但它涉及 NT4 和 2000。
还有一个建议是 使用ESEUTIL 来自 Exchange 安装,但我没有运气让它工作。
有谁知道一种快速、简单的方法来处理这个问题?我说的是在具有 2Gb RAM 的机器上 >50Gb。我计划启动 Visual Studio 并编写一些东西来为我做这件事,但如果有一些已经存在的、稳定的和经过充分测试的东西那就太好了。
[编辑] 我提供了工作 C# 代码来配合接受的答案。
解决方案
最好的选择是只打开原始文件进行读取,打开目标文件进行写入,然后逐块循环复制它。在伪代码中:
f1 = open(filename1);
f2 = open(filename2, "w");
while( !f1.eof() ) {
buffer = f1.read(buffersize);
err = f2.write(buffer, buffersize);
if err != NO_ERROR_CODE
break;
}
f1.close(); f2.close();
[提问者编辑] 好的,这就是 C# 中的样子(速度很慢,但似乎可以正常工作,并且取得了进展):
using System;
using System.Collections.Generic;
using System.IO;
using System.Text;
namespace LoopCopy
{
class Program
{
static void Main(string[] args)
{
if (args.Length != 2)
{
Console.WriteLine(
"Usage: LoopCopy.exe SourceFile DestFile");
return;
}
string srcName = args[0];
string destName = args[1];
FileInfo sourceFile = new FileInfo(srcName);
if (!sourceFile.Exists)
{
Console.WriteLine("Source file {0} does not exist",
srcName);
return;
}
long fileLen = sourceFile.Length;
FileInfo destFile = new FileInfo(destName);
if (destFile.Exists)
{
Console.WriteLine("Destination file {0} already exists",
destName);
return;
}
int buflen = 1024;
byte[] buf = new byte[buflen];
long totalBytesRead = 0;
double pctDone = 0;
string msg = "";
int numReads = 0;
Console.Write("Progress: ");
using (FileStream sourceStream =
new FileStream(srcName, FileMode.Open))
{
using (FileStream destStream =
new FileStream(destName, FileMode.CreateNew))
{
while (true)
{
numReads++;
int bytesRead = sourceStream.Read(buf, 0, buflen);
if (bytesRead == 0) break;
destStream.Write(buf, 0, bytesRead);
totalBytesRead += bytesRead;
if (numReads % 10 == 0)
{
for (int i = 0; i < msg.Length; i++)
{
Console.Write("\b \b");
}
pctDone = (double)
((double)totalBytesRead / (double)fileLen);
msg = string.Format("{0}%",
(int)(pctDone * 100));
Console.Write(msg);
}
if (bytesRead < buflen) break;
}
}
}
for (int i = 0; i < msg.Length; i++)
{
Console.Write("\b \b");
}
Console.WriteLine("100%");
Console.WriteLine("Done");
}
}
}
不隶属于 StackOverflow