【问题标题】:Zip file is getting corrupted after uploaded to server using C#使用 C# 上传到服务器后,Zip 文件损坏
【发布时间】:2013-05-05 22:24:16
【问题描述】:

我正在尝试使用C# (Framework 4)上传一个 zip 文件到服务器,以下是我的代码。

string ftpUrl = ConfigurationManager.AppSettings["ftpAddress"];
string ftpUsername = ConfigurationManager.AppSettings["ftpUsername"];
string ftpPassword = ConfigurationManager.AppSettings["ftpPassword"];  
FtpWebRequest request = (FtpWebRequest)WebRequest.Create(ftpUrl + "Transactions.zip");  
request.Proxy = new WebProxy(); //-----The requested FTP command is not supported when using HTTP proxy.
request.Method = WebRequestMethods.Ftp.UploadFile;
request.Credentials = new NetworkCredential(ftpUsername, ftpPassword);
StreamReader sourceStream = new StreamReader(fileToBeUploaded);
byte[] fileContents = Encoding.UTF8.GetBytes(sourceStream.ReadToEnd());
sourceStream.Close();
request.ContentLength = fileContents.Length;
Stream requestStream = request.GetRequestStream();
requestStream.Write(fileContents, 0, fileContents.Length);
requestStream.Close();
FtpWebResponse response = (FtpWebResponse)request.GetResponse();
Console.WriteLine("Upload File Complete, status {0}", response.StatusDescription);
            response.Close();  

zip 文件上传成功,但是当我尝试从服务器(手动)打开 zip 文件时,它显示Unexpected end of archive 错误。
对于文件压缩,我使用Ionic.zip dll。在传输 zip 文件之前,我能够成功解压。

任何帮助表示赞赏。谢谢。

【问题讨论】:

  • 尽量避免 sourceStream.ReadToEnd() 直接从 sourceStream 复制字节到 requestStream

标签: c# ftp dotnetzip


【解决方案1】:

这就是问题所在:

StreamReader sourceStream = new StreamReader(fileToBeUploaded);
byte[] fileContents = Encoding.UTF8.GetBytes(sourceStream.ReadToEnd());

StreamReader(和任何TextReader)用于文本数据。 zip 文件不是文本数据。

只需使用:

byte[] fileContents = File.ReadAllBytes(fileToBeUploaded);

这样您就不会将二进制数据视为文本,因此它不应该被破坏。

或者,不要单独将其全部加载到内存中 - 只需流式传输数据:

using (var requestStream = request.GetRequestStream())
{
    using (var input = File.OpenRead(fileToBeUploaded))
    {
        input.CopyTo(requestStream);
    }
}

另外请注意,您应该对所有这些流使用using 语句,而不是仅仅调用Close - 这样即使抛出异常,资源也会被释放。

【讨论】:

  • 有没有办法知道我们的文件是否上传成功,比如status.code =OK
  • @user1671639:是的 - 从FtpWebResponse 获取状态码,看看它是CommandOK 还是FileActionOK。 (可能还有其他一些需要寻找 - 尝试一下。)老实说,如果它失败了,我希望它会在你收到响应时抛出异常。
  • 我尝试指定 FtpWebResponse response = (FtpWebResponse)request.GetResponse(); 并从 response.StatusDescription = "226 File receive OK." 得到它。但我已经按照您的建议使用了using 声明。这会影响我的计划吗?
  • @user1671639:我不明白你的意思。是的,您应该使用 using 语句作为响应...但我建议使用状态 代码 而不是描述。
  • 我的“*.bin”正在损坏。我试过你的方法,但没有帮助。任何帮助或猜测。谢谢