C#异步调用的好处和方法分享
程序员文章站
2023-12-01 22:21:46
异步方法很好的解决了这些问题,异步执行某个方法,程序立即开辟一个新线程去运行你的方法,主线程包括界面就不会死掉了。异步如何开始,好理解,现在我们讨论的是如何结束这个异步出来...
异步方法很好的解决了这些问题,异步执行某个方法,程序立即开辟一个新线程去运行你的方法,主线程包括界面就不会死掉了。异步如何开始,好理解,现在我们讨论的是如何结束这个异步出来的新线程。
首先,异步出来的新线程,必须回收,不回收是浪费资源的可耻行为,.net也是不允许的,所以你别想钻空子,俗话说,请神容易送神难,就是这个道理。下面你可以很容易想到,回收分为2种情况:主动回收和被动回收(当然,这是我自己的理解,微软可不是这么说的),主动回收就是,你去监视那个线程,并且等待,当异步方法完成了,就把异步线程回收,焦点回归主线程,实际上就是上篇文章《c#异步初步》的那种情况,begininvoke之后又endinvoke,如果在endinvoke的时候,该异步线程没有完成操作,那么整个程序,包括主线程,又在阻塞了,又会出现界面“死”的情况。要想解决这个问题,就使用“被动回收”方式,其中一个重要的办法就是“异步回调”。 核心有二: a、 用回调函数(本例中为callbackmethod),异步结束后,自动调用此回调函数。 b、 而不在主线程中手工等待异步结束,如上两例中在主线程中调用endinvoke。此种方法,是在回调函数中调用endinvoke的。 异步回调的大概流程是这样的:首先启动异步,启动参数加上异步结束时执行的方法,然后这个异步线程就不用管了,最后当这个异步线程自己完成工作了,就自动执行启动参数里的那个方法,这样确实很省心,可是代码写起来,就很复杂了。 下面是搜藏的代码:
//首先准备好,要进行异步的方法(能异步的,最好不多线程)
privatestringmethodname(intnum,outintnum2)
{
num2=num;
return"helloworld";
}
//程序终点
//异步完成时,执行的方法(回调方法),此方法只能有iasyncresult一个参数,但是该参数几乎万能,可以传递object
privatevoidcallbackmethod(iasyncresultar)
{
//从异步状态ar.asyncstate中,获取委托对象
delegatenamedn=(delegatename)ar.asyncstate;
//输出参数
inti;
//一定要endinvoke,否则你的下场很惨
stringr=dn.endinvoke(outi,ar);
messagebox.show("异步完成喽!i的值是"i.tostring()",r的值是"r);
}
//定义与方法同签名的委托
privatedelegatestringdelegatename(intnum,outintnum2);
//程序入口
privatevoidrun()
{
//实例化委托并初赋值
delegatenamedn=newdelegatename(methodname);
//输出参数
inti;
//实例化回调方法
//把asynccallback看成delegate你就懂了,实际上asynccallback是一种特殊的delegate,就像event似的
asynccallbackacb=newasynccallback(callbackmethod);
//异步开始
//如果参数acb换成null则表示没有回调方法
//最后一个参数dn的地方,可以换成任意对象,该对象可以被回调方法从参数中获取出来,写成null也可以。参数dn相当于该线程的id,如果有多个异步线程,可以都是null,但是绝对不能一样,不能是同一个object,否则异常
iasyncresultiar=dn.begininvoke(1,outi,acb,dn);
//去做别的事
//…………
}
//最后的结果应该是:i=1,r="helloworld"
//另外,如果可以,定义委托的时候可以选择不用过多的修饰:
///<summary>
///定义委托
///</summary>
///<returns></returns>
publicdelegateboolasyncdelegate();
///<summary>
///callbackmethodmusthavethesamesignatureasthe
///asynccallbackdelegate
///</summary>
///<paramname="ar"></param>
privatevoidcallbackmethod(iasyncresultar)
{
//retrievethedelegate.
asyncdelegatedlgt=(asyncdelegate)ar.asyncstate;
//callendinvoketoretrievetheresults.
dlgt.endinvoke(ar);
}
//其他方法中调用:
//异步执行
//指定委托方法
asyncdelegateisgt=newasyncdelegate(icpinfo.insert);
iasyncresultar=isgt.begininvoke(newasynccallback(callbackmethod),isgt);
首先,异步出来的新线程,必须回收,不回收是浪费资源的可耻行为,.net也是不允许的,所以你别想钻空子,俗话说,请神容易送神难,就是这个道理。下面你可以很容易想到,回收分为2种情况:主动回收和被动回收(当然,这是我自己的理解,微软可不是这么说的),主动回收就是,你去监视那个线程,并且等待,当异步方法完成了,就把异步线程回收,焦点回归主线程,实际上就是上篇文章《c#异步初步》的那种情况,begininvoke之后又endinvoke,如果在endinvoke的时候,该异步线程没有完成操作,那么整个程序,包括主线程,又在阻塞了,又会出现界面“死”的情况。要想解决这个问题,就使用“被动回收”方式,其中一个重要的办法就是“异步回调”。 核心有二: a、 用回调函数(本例中为callbackmethod),异步结束后,自动调用此回调函数。 b、 而不在主线程中手工等待异步结束,如上两例中在主线程中调用endinvoke。此种方法,是在回调函数中调用endinvoke的。 异步回调的大概流程是这样的:首先启动异步,启动参数加上异步结束时执行的方法,然后这个异步线程就不用管了,最后当这个异步线程自己完成工作了,就自动执行启动参数里的那个方法,这样确实很省心,可是代码写起来,就很复杂了。 下面是搜藏的代码:
复制代码 代码如下:
//首先准备好,要进行异步的方法(能异步的,最好不多线程)
privatestringmethodname(intnum,outintnum2)
{
num2=num;
return"helloworld";
}
//程序终点
//异步完成时,执行的方法(回调方法),此方法只能有iasyncresult一个参数,但是该参数几乎万能,可以传递object
privatevoidcallbackmethod(iasyncresultar)
{
//从异步状态ar.asyncstate中,获取委托对象
delegatenamedn=(delegatename)ar.asyncstate;
//输出参数
inti;
//一定要endinvoke,否则你的下场很惨
stringr=dn.endinvoke(outi,ar);
messagebox.show("异步完成喽!i的值是"i.tostring()",r的值是"r);
}
//定义与方法同签名的委托
privatedelegatestringdelegatename(intnum,outintnum2);
//程序入口
privatevoidrun()
{
//实例化委托并初赋值
delegatenamedn=newdelegatename(methodname);
//输出参数
inti;
//实例化回调方法
//把asynccallback看成delegate你就懂了,实际上asynccallback是一种特殊的delegate,就像event似的
asynccallbackacb=newasynccallback(callbackmethod);
//异步开始
//如果参数acb换成null则表示没有回调方法
//最后一个参数dn的地方,可以换成任意对象,该对象可以被回调方法从参数中获取出来,写成null也可以。参数dn相当于该线程的id,如果有多个异步线程,可以都是null,但是绝对不能一样,不能是同一个object,否则异常
iasyncresultiar=dn.begininvoke(1,outi,acb,dn);
//去做别的事
//…………
}
//最后的结果应该是:i=1,r="helloworld"
//另外,如果可以,定义委托的时候可以选择不用过多的修饰:
///<summary>
///定义委托
///</summary>
///<returns></returns>
publicdelegateboolasyncdelegate();
///<summary>
///callbackmethodmusthavethesamesignatureasthe
///asynccallbackdelegate
///</summary>
///<paramname="ar"></param>
privatevoidcallbackmethod(iasyncresultar)
{
//retrievethedelegate.
asyncdelegatedlgt=(asyncdelegate)ar.asyncstate;
//callendinvoketoretrievetheresults.
dlgt.endinvoke(ar);
}
//其他方法中调用:
//异步执行
//指定委托方法
asyncdelegateisgt=newasyncdelegate(icpinfo.insert);
iasyncresultar=isgt.begininvoke(newasynccallback(callbackmethod),isgt);
上一篇: Bootstrap 3 按钮标签实例代码