如何在没有 CopyFile 或 CopyFileEx 的情况下在 Windows 上复制大文件?

StackOverflow https://stackoverflow.com/questions/92114

  •  01-07-2019
  •  | 
  •  

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");
        }
    }
}

其他提示

如果您想编写代码,可以优化的一种方法是以块的形式发送文件(例如使用 MTOM)。我使用这种方法将巨大的文件从数据中心发送到我们的办公室进行打印。

另外,检查提到的 TeraCopy 实用程序 这里..

许可以下: CC-BY-SA归因
不隶属于 StackOverflow
scroll top