重用FtpWebRequest

问题描述 投票:9回答:3

我正在尝试使用FtpWebRequestWebRequestMethods.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();

我希望有人能解释这是如何工作的。预先感谢。

c# .net ftp ftpwebrequest
3个回答
7
投票

我不认为这会得到回答,所以我通过告诉您如何解决来“结束”。

嗯,我并没有真正解决它。但是,我确实通过重新创建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以重新使用,请回答。但是至少我知道没有多余的信息正在传输。

感谢所有感兴趣并花时间思考答案的人。


3
投票

FtpWebRequest只能用于1个请求,例如获取文件大小或下载文件,但不能同时使用。您必须创建2个FtpWebRequests。在后台,FtpWebRequest会注意到它是相同的URL和凭据,并且只要IsKeepAlieve为true(这是默认设置),它将在不关闭它的情况下重用相同的ftp连接。

这是Microsoft不良设计的可悲示例。他们不想让我们显式打开和关闭连接,而是希望为我们自动进行连接并使所有人感到困惑。


0
投票

您可能要使用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");

源也在那里,因此您可以将一些事件附加到读取过程中,以进行下载进度跟踪。

© www.soinside.com 2019 - 2024. All rights reserved.