C# 多线程 用委托实现异步_调用委托的BeginInvoke和EndInvoke方法

1.C#中的每一个委托都内置了BeginInvoke和EndInvoke方法,如果委托的方法列表里只有一个方法,那么这个方法就可以异步执行(不在当前线程里执行,另开辟一个线程执行)。委托的BeginInvoke和EndInvoke方法就是为了上述目的而生的。

2.原始线程发起了一个异步线程,有如下三种执行方式:

方式一:等待一直到完成,即原始线程在发起了异步线程以及做了一些必要处理之后,原始线程就中断并等待异步线程结束再继续执行。

方式二:轮询,即原始线程定期检查发起的线程是否完成,如果没有则可以继续做一些其它事情。

方式三:回调,即原始线程一直执行,无需等待或检查发起的线程是否完成。在发起的线程执行结束,发起的线程就会调用用户定义好的回调方法,由这个回调方法在调用EndInvoke之前处理异步方法执行得到的结果。

3.一个控制台小程序,使用了上面三种方式,执行结果如下:

4.代码:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.Remoting.Messaging;
using System.Text;
using System.Threading;

namespace 用委托实现异步_调用BeginInvoke和EndInvoke方法
{

    delegate long MyDel(int first,int second); //声明委托类型

    class Program
    {
        //声明委托类型的方法
        static long Sum(int x,int y)
        {
            Console.WriteLine("                    Inside Sum");
            Thread.Sleep(200);
            return x + y;
        }

        //定义当异步线程执行结束要执行的回调函数
        static void CallWhenDone(IAsyncResult iar)
        {
            Console.WriteLine("                    Inside CallWhenDone");
            AsyncResult ar = (AsyncResult)iar;
            MyDel del = (MyDel)ar.AsyncDelegate;

            long result = del.EndInvoke(iar);
            Console.WriteLine("                    The result is {0}.", result);
        }

        //方式一:等待异步线程结束,再继续执行主线程
        static void WaitUntilDoneStyle()
        {
            MyDel del = new MyDel(Sum);
            Console.WriteLine("Before BeginInvoke");
            IAsyncResult iar = del.BeginInvoke(3, 5, null, null); //开始异步调用
            Console.WriteLine("After BeginInvoke");

            Console.WriteLine("Doing main stuff before");
            long result = del.EndInvoke(iar); //等待异步线程结束并获取结果
            Console.WriteLine("After EndInvoke:{0}", result);
            Console.WriteLine("Doing main stuff after");
        }

        //方式二:轮询检查异步线程是否结束,若没结束则执行主线程
        static void LunXunPollingStyle()
        {
            MyDel del = new MyDel(Sum);
            Console.WriteLine("Before BeginInvoke");
            IAsyncResult iar = del.BeginInvoke(3, 5, null, null); //开始异步调用
            Console.WriteLine("After BeginInvoke");

            while (!iar.IsCompleted)
            {
                Console.WriteLine("Not Done.Doing main stuff");
                //继续处理主线程事情
                for (long i = 0; i < 10000000; i++)
                    ;
            }
            Console.WriteLine("Done");
            long result = del.EndInvoke(iar); //调用EndInvoke来获取结果并进行清理
            Console.WriteLine("Result: {0}", result);
        }

        //方式三:回调方式,当异步线程结束,系统调用用户自定义的方法来处理结果(包括调用委托的EndInvoke方法)
        static void CallBackStyle()
        {
            MyDel del = new MyDel(Sum);
            Console.WriteLine("Before BeginInvoke");
            IAsyncResult iar = del.BeginInvoke(3, 5, new AsyncCallback(CallWhenDone), null);
            Console.WriteLine("After BeginInvoke");
            Console.WriteLine("Doing more work in main.");
            Thread.Sleep(500);
            Console.WriteLine("Done with Main. Exiting.");
        }

        static void Main(string[] args)
        {
            //方式一:等待异步线程结束,再继续执行主线程
            Console.WriteLine();
            Console.WriteLine("--------方式一:等待异步线程结束,再继续执行主线程--------");
            WaitUntilDoneStyle();

            //方式二:轮询检查异步线程是否结束,若没结束则执行主线程
            Console.WriteLine();
            Console.WriteLine("--------方式二:轮询检查异步线程是否结束,若没结束则执行主线程--------");
            LunXunPollingStyle();

            //方式三:回调方式,当异步线程结束,系统调用用户自定义的方法来处理结果(包括调用委托的EndInvoke方法)
            Console.WriteLine();
            Console.WriteLine("--------方式三:回调方式,当异步线程结束,系统调用用户自定义的方法来处理结果(包括调用委托的EndInvoke方法)--------");
            CallBackStyle();
        }

    }
}
时间: 2024-12-25 08:33:54

C# 多线程 用委托实现异步_调用委托的BeginInvoke和EndInvoke方法的相关文章

C#用委托(Delegate)的BeginInvoke和EndInvoke方法操作线程

C#用委托(Delegate)的BeginInvoke和EndInvoke方法操作线程C# 2011-03-05 13:06:24 阅读19 评论0   字号:大中小 订阅 用委托(Delegate)的BeginInvoke和EndInvoke方法操作线程 在C#中使用线程的方法很多,使用委托的BeginInvoke和EndInvoke方法就是其中之一. BeginInvoke方法可以使用线程异步地执行委托所指向的方法.然后通过EndInvoke方法获得方法的返回值(EndInvoke方法的返回

C#多线程编程之:异步事件调用

当一个事件被触发时,订阅该事件的方法将在触发该事件的线程中执行.也就是说,订阅该事件的方法在触发事件的线程中同步执行.由此,存在一个问 题:如果订阅事件的方法执行时间很长,触发事件的线程被阻塞,长时间等待方法执行完毕.这样,不仅影响后续订阅事件方法的执行,也影响主线程及时响应用户 的其他请求.如何处理这个问题呢?讲到此,我想您已经想到了,那就是异步事件调用. 怎样实现异步事件调用呢?如果您对事件比较了解的话,您应该知道事件的本质其实是一种MulticastDelegate(多播委托). Mult

异步使用委托delegate --- BeginInvoke和EndInvoke方法

当我们定义一个委托的时候,一般语言运行时会自动帮委托定义BeginInvoke 和 EndInvoke两个方法,这两个方法的作用是可以异步调用委托. 方法BeginInvoke有两个参数: AsyncCallBack:回调函数,是一个委托,没有返回值,可以传一个参数,参数类型是object: object AsyncState :回调函数的参数. BeginInvoke的返回值是IAsyncResult, 方法EndInvoke需要的参数是BeginInvoke的返回值IAsyncResult.

[C#学习笔记之异步编程模式2]BeginInvoke和EndInvoke方法 (转载)

为什么要进行异步回调?众所周知,普通方法运行,是单线程的,如果中途有大型操作(如:读取大文件,大批量操作数据库,网络传输等),都会导致方法阻塞,表现在界面上就是,程序卡或者死掉,界面元素不动了,不响应了.异步方法很好的解决了这些问题,异步执行某个方法,程序立即开辟一个新线程去运行你的方法,主线程包括界面就不会死掉了.异步调用并不是要减少线程的开销, 它的主要目的是让调用方法的主线程不需要同步等待在这个函数调用上, 从而可以让主线程继续执行它下面的代码. BeginInvoke方法可以使用线程异步

c# 委托与异步调用

背景:在winform UI中,有时需要对控件进行比较频繁的刷新,如进度条.picturebox显示视频等.如果在主线程进行这些刷新操作,操作还未完成就将执行下一次刷新,程序将发生错误:如果只是创建另一个线程执行这些操作,将和主线程产生竞争,造成界面锁死(因此windows GUI编程有一个规则,就是只能通过创建控件的线程来操作控件的数据,否则就可能产生不可预料的结果).这时候,我们就可以用委托与异步来解决这个问题. 委托:回顾一下委托 ,一.定义委托,委托定义的参数与传递给委托的方法的参数一致

.NET委托解析(异步委托)

上一篇我们了解到了,委托的基本感念,列举了几个委托的实例,并根据实例来反编译源码查看.NET 委托的内部实现,从浅入深的角度来详细的去解析委托的实质,本文将系上篇继续讨论异步委托的实现以及异步委托的源码解析. 首先本文只会从委托的层面的去编写,不会涉及到深层次的异步.(后续的系列中将会对异步进行深入讲解.敬请关注.). 委托的异步调用方式 在上一篇中我们在实例中是直接通过委托对象的,例如: private static void Main(string[] args)  {        Pro

C#委托的异步调用

#委托的异步调用本文将主要通过"同步调用"."异步调用"."异步回调"三个示例来讲解在用委托执行同一个"加法类"的时候的的区别和利弊.   首先,通过代码定义一个委托和下面三个示例将要调用的方法: /*添加的命名空间using System.Threading;using System.Runtime.Remoting.Messaging;*/    public delegate int AddHandler(int a,i

C# 委托的三种调用示例(同步调用 异步调用 异步回调)

本文将主要通过同步调用.异步调用.异步回调三个示例来讲解在用委托执行同一个加法类的时候的的区别和利弊 首先,通过代码定义一个委托和下面三个示例将要调用的方法: public delegate int AddHandler(int a,int b); public class 加法类 { public static int Add(int a, int b) { Console.WriteLine("开始计算:" + a + "+" + b); Thread.Slee

C#委托的异步调用1

本文将主要通过“同步调用”.“异步调用”.“异步回调”三个示例来讲解在用委托执行同一个“加法类”的时候的的区别和利弊. 首先,通过代码定义一个委托和下面三个示例将要调用的方法: /*添加的命名空间using System.Threading;using System.Runtime.Remoting.Messaging;*/    public delegate int AddHandler(int a,int b);    public class 加法类    {        public