我正在尝试使用FtpWebRequest
和WebRequestMethods.Ftp.DownloadFile
的方法来制作从FTP下载文件的简单方法。问题是我无法显示下载进度,因此需要提前知道文件大小才能计算出传输的百分比。但是,当我在GetResponse
中调用FtpWebRequest
时,ContentLength
成员为-1。
[确定-这样我可以使用WebRequestMethods.Ftp.GetFileSize
方法预先获取文件的大小。没问题。然后在获得大小后,我下载文件。
这是出现问题的地方...
获得大小后,我尝试重用FtpWebRequest
并将方法重置为WebRequestMethods.Ftp.DownloadFile
。这会导致System.InvalidOperationException
发出类似“发送请求后无法执行此操作”之类的信息。 (可能不是确切的提法-从我用瑞典语得到的翻译而来)。
我在其他地方发现,只要将KeepAlive
属性设置为true,就没关系,连接保持活动状态。这是我不了解的...我创建的唯一对象是我的FtpWebRequest
对象。如果我创建另一个,它怎么知道要使用什么连接?还有什么凭证?
伪代码:
Create FtpWebRequest
Set Method property to GetFileSize
Set KeepAlive property to true
Set Credentials property to new NetworkCredential(...)
Get FtpWebResponse from the request
Read and store ContentLength
现在我得到了文件大小。现在该下载文件了。现在,设置方法将导致上述异常。那我要创建一个新的FtpWebRequest
吗?还是无论如何都要重置要重用的请求? (关闭响应没有区别。)
我不了解如何重新创建对象。我可以这样做,但是感觉不对。所以我在这里发布,希望找到正确的方法。
这是(无效的)代码(输入为sURI,sDiskName,sUser和sPwd。):
FtpWebRequest request = (FtpWebRequest)FtpWebRequest.Create(sURI);
request.Method = WebRequestMethods.Ftp.GetFileSize;
request.Credentials = new NetworkCredential(sUser, sPwd);
request.UseBinary = true;
request.UsePassive = true;
request.KeepAlive = true;
FtpWebResponse resp = (FtpWebResponse)request.GetResponse();
int contLen = (int)resp.ContentLength;
resp.Close();
request.Method = WebRequestMethods.Ftp.DownloadFile;
resp = (FtpWebResponse)request.GetResponse();
Stream inStr = resp.GetResponseStream();
byte[] buff = new byte[16384];
sDiskName = Environment.ExpandEnvironmentVariables(sDiskName);
FileStream file = File.Create(sDiskName);
int readBytesCount;
int readTotal=0;
while ((readBytesCount = inStr.Read(buff, 0, buff.Length)) > 0)
{
readTotal += readBytesCount;
toolStripProgressBar1.Value = 100*readTotal/contLen;
Application.DoEvents();
file.Write(buff, 0, readBytesCount);
}
file.Close();
我希望有人能解释这是如何工作的。预先感谢。
我不认为这会得到回答,所以我通过告诉您如何解决来“结束”。
嗯,我并没有真正解决它。但是,我确实通过重新创建FtpWebRequest
来测试下载,并注意到在FTP服务器上它的行为符合我的要求,即仅登录一次,然后依次执行我的请求。
这是获取文件大小并开始下载的代码的结尾:
// Start by fetching the file size
FtpWebRequest request = (FtpWebRequest)FtpWebRequest.Create(sURI);
request.Method = WebRequestMethods.Ftp.GetFileSize;
NetworkCredential nc = new NetworkCredential(sUser, sPwd);
request.Credentials = nc;
request.UseBinary = true;
request.UsePassive = true;
request.KeepAlive = true;
// Get the result (size)
FtpWebResponse resp = (FtpWebResponse)request.GetResponse();
Int64 contLen = resp.ContentLength;
// and now download the file
request = (FtpWebRequest)FtpWebRequest.Create(sURI);
request.Method = WebRequestMethods.Ftp.DownloadFile;
request.Credentials = nc;
request.UseBinary = true;
request.UsePassive = true;
request.KeepAlive = true;
resp = (FtpWebResponse)request.GetResponse();
因此,如果可以重设FtpWebRequest
以重新使用,请回答。但是至少我知道没有多余的信息正在传输。
感谢所有感兴趣并花时间思考答案的人。
FtpWebRequest只能用于1个请求,例如获取文件大小或下载文件,但不能同时使用。您必须创建2个FtpWebRequests。在后台,FtpWebRequest会注意到它是相同的URL和凭据,并且只要IsKeepAlieve为true(这是默认设置),它将在不关闭它的情况下重用相同的ftp连接。
这是Microsoft不良设计的可悲示例。他们不想让我们显式打开和关闭连接,而是希望为我们自动进行连接并使所有人感到困惑。
您可能要使用Async方法。这是MSDN文档的链接。
http://msdn.microsoft.com/en-us/library/system.net.ftpwebrequest.aspx
GetResponseAsync()
这也将使您的应用程序也不会锁定,因此您不必使用
Application.DoEvents();
您也可以考虑使用替代的ftp库。 imo FtpWebRequest并不是最好的ftp类。快速搜索打开了该库。 Ftp并非像HTTP那样是无状态的。我更喜欢允许您创建客户端,打开连接并保持连接活动的库。
http://sanity-free.org/dist/NullFX.Net-binary.zip
这是我找到的代码示例
FtpClient client =
new FtpClient(
new IPEndPoint( IPAddress.Loopback, 21 ),
new NetworkCredential( "test", "testing@localdomain" )
);
client.Connect();
client.Download("testfile.zip", @"C:\downloads\testfile.zip");
源也在那里,因此您可以将一些事件附加到读取过程中,以进行下载进度跟踪。