使用HttpClient异步方法的BackgroundWorker
本文关键字:BackgroundWorker 异步方法 HttpClient 使用 | 更新日期: 2023-09-27 18:26:18
我过去曾在Windows窗体应用程序中使用过BackgroundWorker
。对于我的新练习,我需要在工人内部使用async
方法,对此我有点困惑。
这是我的代码结构。在表单加载事件中,我正在创建BackgroundWorker
对象和设置事件
private void fMain_Load( object sender, EventArgs e ) {
bw = new BackgroundWorker();
bw.WorkerReportsProgress = true;
bw.DoWork += new DoWorkEventHandler( bw_DoWork );
bw.ProgressChanged += new ProgressChangedEventHandler( bw_ProgressChanged );
bw.RunWorkerCompleted += new RunWorkerCompletedEventHandler( bw_RunWorkerCompleted );
}
当用户点击按钮时,我正在启动工作
private void btnGenerate_Click( object sender, EventArgs e ) {
Settings settings = new Settings();
pbCounter.Visible = true;
btnGenerate.Enabled = false;
bw.RunWorkerAsync( settings );
}
这是工人代码
private async void bw_DoWork( object sender, DoWorkEventArgs e ) {
try {
for ( int ix = 1; i <= 100; i++ ) {
using (var client = new HttpClient()) {
[???? how to call and wait here ????]
HttpResponseMessage response = await client.PostAsync( "endpoint", new StringContent( JsonConvert.SerializeObject( formContent ), Encoding.UTF8, "application/json" ) );
}
//The counter will keep track of your process
Application.DoEvents();
int percentage = ix * 100 / settings.TotalRuns;
bw.ReportProgress( percentage );
}
}
catch ( Exception ex ) {
MessageBox.Show( ex.Message, "Gift Creator", MessageBoxButtons.OK, MessageBoxIcon.Error );
}
}
如果使用async
和await
,则不需要后台工作人员。事实上,您的后台工作人员不会工作,因为当您使用await
时,控制权会返回给调用者。当DoWork
处理程序将控制权返回给它的调用者时,后台工作程序将终止而不继续它的重设任务。
所以我会制作按钮处理程序async
,并在那里进行http请求:
private async void btnGenerate_Click(object sender, EventArgs e)
{
const int totalRuns = 5;
pbCounter.Visible = true;
pbCounter.Minimum = 0;
pbCounter.Maximum = totalRuns;
pbCounter.Value = 0;
btnGenerate.Enabled = false;
try
{
for ( int i = 1; i <= totalRuns; i++ )
{
using (var client = new HttpClient())
await client.PostAsync( "endpoint", new StringContent( JsonConvert.SerializeObject( formContent ), Encoding.UTF8, "application/json" ) );
pbCounter.Value = i;
}
}
catch (Exception ex )
{
MessageBox.Show( ex.Message, "Gift Creator", MessageBoxButtons.OK, MessageBoxIcon.Error );
}
btnGenerate.Enabled = true;
}
因此,这个处理程序在等待http请求时将控制权交还给调用方(可以说过于简化了:UI)。请求完成后,将在pbCounter.Value = i
行和UI线程上继续执行!因此,您可以安全地更新进度条,因为您不是从另一个线程进行更新的。
我希望这对你有帮助。请注意,我将pbCounter.Maximum
设置为要运行的循环数,因此不需要计算百分比。