我有一个组件,它在每个单独的线程中处理多个Web请求.每个WebRequest处理都是同步的.
public class WebRequestProcessor:System.ComponentModel.Component
{
List<Worker> tlist = new List<Worker>();
public void Start()
{
foreach(string url in urlList){
// Create the thread object. This does not start the thread.
Worker workerObject = new Worker();
Thread workerThread = new Thread(workerObject.DoWork);
// Start the worker thread.
workerThread.Start(url);
tlist.Add(workerThread);
}
}
}
public class Worker
{
// This method will be called when the thread is started.
public void DoWork(string url)
{
// prepare the web page we will be asking for
HttpWebRequest request = (HttpWebRequest)
WebRequest.Create(url);
// execute the request
HttpWebResponse response = (HttpWebResponse)
request.GetResponse();
// we will read data via the response stream
Stream resStream = response.GetResponseStream();
// process stream
}
}
Run Code Online (Sandbox Code Playgroud)
现在我必须找到如何取消所有请求的最佳方法.
一种方法是将每个同步WebRequest转换为异步,并使用WebRequest.Abort取消处理.
另一种方法是释放线程指针并允许所有线程使用GC死亡.
svi*_*ick 10
如果要下载1000个文件,一次启动1000个线程肯定不是最佳选择.与一次只下载几个文件相比,它不仅可能不会让你获得任何加速,它还需要至少1 GB的虚拟内存.创建线程很昂贵,尽量避免在循环中这样做.
您应该做的是Parallel.ForEach()与请求和响应操作的异步版本一起使用.例如像这样(WPF代码):
private void Start_Click(object sender, RoutedEventArgs e)
{
m_tokenSource = new CancellationTokenSource();
var urls = …;
Task.Factory.StartNew(() => Start(urls, m_tokenSource.Token), m_tokenSource.Token);
}
private void Cancel_Click(object sender, RoutedEventArgs e)
{
m_tokenSource.Cancel();
}
void Start(IEnumerable<string> urlList, CancellationToken token)
{
Parallel.ForEach(urlList, new ParallelOptions { CancellationToken = token },
url => DownloadOne(url, token));
}
void DownloadOne(string url, CancellationToken token)
{
ReportStart(url);
try
{
var request = WebRequest.Create(url);
var asyncResult = request.BeginGetResponse(null, null);
WaitHandle.WaitAny(new[] { asyncResult.AsyncWaitHandle, token.WaitHandle });
if (token.IsCancellationRequested)
{
request.Abort();
return;
}
var response = request.EndGetResponse(asyncResult);
using (var stream = response.GetResponseStream())
{
byte[] bytes = new byte[4096];
while (true)
{
asyncResult = stream.BeginRead(bytes, 0, bytes.Length, null, null);
WaitHandle.WaitAny(new[] { asyncResult.AsyncWaitHandle,
token.WaitHandle });
if (token.IsCancellationRequested)
break;
var read = stream.EndRead(asyncResult);
if (read == 0)
break;
// do something with the downloaded bytes
}
}
response.Close();
}
finally
{
ReportFinish(url);
}
}
Run Code Online (Sandbox Code Playgroud)
这样,当您取消操作时,将取消所有下载并且不会启动任何新下载.此外,您可能需要设置MaxDegreeOfParallelism的ParallelOptions,所以你是不是同时做太多的下载.
我不确定您要对要下载的文件做什么,所以使用StreamReader可能是更好的选择.
| 归档时间: |
|
| 查看次数: |
5649 次 |
| 最近记录: |