Jus*_*mes 2 c# asynchronous web-services asmx async-await
我正在使用asmx服务并尝试在尝试调用时执行async/await
目前我的控制器如下
Public ActionResult Index()
{
var data = _repository.GetDataFromAsmxService(somedata);
}
Run Code Online (Sandbox Code Playgroud)
// RepositoryClass方法
public List<ObjectToReturn> GetDataFromAsmxService(somedata)
{
var res = _asmxService.GetReportData(somedata);
var result = process(res);
return result;
}
Run Code Online (Sandbox Code Playgroud)
在进一步查看生成的refrence.cs文件后,发现了等效调用的void async和onCompleted方法.
所以我把它改成了
private void GetReportDataCallBack(object sender, GetReportDataCompletedEventArgs e)
{
var res = e.Result;
var result = process(res);
}
public List<ObjectToReturn> GetDataFromAsmxService(somedata)
{
_asmxService.GetReportDataCompleted +=GetReportDataCallBack;
_asmxService.GetReportDataAsync(somedata);
}
Run Code Online (Sandbox Code Playgroud)
但现在回调发生在Event委托上,所以我的控制器期望返回数据不会得到它. 什么是从控制器调用async/await并从asmx webservice获取数据的更好方法
代码来自Refenrece.cs //生成的代码
[System.Web.Services.Protocols.SoapHeaderAttribute("ApiAuthenticationValue")]
[System.Web.Services.Protocols.SoapDocumentMethodAttribute("http://webservice.com/GetReportData", Use=System.Web.Services.Description.SoapBindingUse.Literal, ParameterStyle=System.Web.Services.Protocols.SoapParameterStyle.Bare)]
[return: System.Xml.Serialization.XmlElementAttribute(Namespace="http://webservice.com/")]
public GetReportDataResponse GetReportData([System.Xml.Serialization.XmlElementAttribute(Namespace="http://webservice.com/")] GetReportDataRequest GetReportDataRequest) {
object[] results = this.Invoke("GetReportData", new object[] {
GetReportDataRequest});
return ((GetReportDataResponse)(results[0]));
}
/// <remarks/>
public void GetReportDataAsync(GetReportDataRequest GetReportDataRequest) {
this.GetReportDataAsync(GetReportDataRequest, null);
}
/// <remarks/>
public void GetReportDataAsync(GetReportDataRequest GetReportDataRequest, object userState) {
if ((this.GetReportDataOperationCompleted == null)) {
this.GetReportDataOperationCompleted = new System.Threading.SendOrPostCallback(this.OnGetReportDataOperationCompleted);
}
this.InvokeAsync("GetReportData", new object[] {
GetReportDataRequest}, this.GetReportDataOperationCompleted, userState);
}
private void OnGetReportDataOperationCompleted(object arg) {
if ((this.GetReportDataCompleted != null)) {
System.Web.Services.Protocols.InvokeCompletedEventArgs invokeArgs = ((System.Web.Services.Protocols.InvokeCompletedEventArgs)(arg));
this.GetReportDataCompleted(this, new GetReportDataCompletedEventArgs(invokeArgs.Results, invokeArgs.Error, invokeArgs.Cancelled, invokeArgs.UserState));
}
}
Run Code Online (Sandbox Code Playgroud)
你所拥有的是基于事件的异步模式(EAP),你想要做的就是将它包装在一个任务中,这样你就可以等待它.你这样做是通过TaskCompletionSource.
public async Task<List<ObjectToReturn>> GetDataFromAsmxServiceAsync(GetReportDataRequest somedata)
{
var tcs = new TaskCompletionSource<GetReportDataResponse>();
_asmxService.GetReportDataCompleted += GetReportDataCallBack;
_asmxService.GetReportDataAsync(somedata, tcs); //we pass tcs in so it can be used from the callback.
GetReportDataResponse res;
try
{
res = await tcs.Task;
}
finally
{
//unsubscribe from the handler when done so we don't get a leak.
_asmxService.GetReportDataCompleted -= GetReportDataCallBack;
}
var result = process(res);
return result;
}
private void GetReportDataCallBack(object sender, GetReportDataCompletedEventArgs e)
{
var tcs = (TaskCompletionSource<GetReportDataResponse>)e.UserState;
if (e.Cancelled)
{
tcs.TrySetCanceled();
}
else if (e.Error != null)
{
tcs.TrySetException(e.Error);
}
else
{
tcs.TrySetResult(e.Result);
}
}
Run Code Online (Sandbox Code Playgroud)
您还需要将控制器更改为异步
public Task<ActionResult> IndexAsync()
{
var data = await _repository.GetDataFromAsmxServiceAsync(somedata);
}
Run Code Online (Sandbox Code Playgroud)
这是一篇很好的文章,解释了如何从同步MVC代码转换为异步MVC代码,并解释了为什么即使Index现在调用IndexAsync它仍然可以作为索引的控制器.
| 归档时间: |
|
| 查看次数: |
3757 次 |
| 最近记录: |