.NET异步编程总结----四种实现模式

createh52个月前 (02-04)技术教程10

第一种方法:BeginEnvoke EndEnvoke方法,属于“等待”类。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading;

namespace 异步调用实现方法汇总
{
    /// 
    /// 异步调用方法总结:
    /// 1.BeginEnvoke EndEnvoke
    /// 当使用BeginInvoke异步调用方法时,如果方法未执行完,EndInvoke方法就会一直阻塞,直到被调用的方法执行完毕
    /// 
    class Program
    {
        public delegate void PrintDelegate(string s);
        static void Main(string[] args)
        {
            PrintDelegate printDelegate = Print;
            Console.WriteLine("主线程");

            IAsyncResult result= printDelegate.BeginInvoke("Hello World.", null, null);
            Console.WriteLine("主线程继续执行...");
            //当使用BeginInvoke异步调用方法时,如果方法未执行完,EndInvoke方法就会一直阻塞,直到被调用的方法执行完毕
            printDelegate.EndInvoke(result);

            Console.WriteLine("Press any key to continue...");
            Console.ReadKey(true);
        }

        public static void Print(string s)
        {
            Console.WriteLine("异步线程开始执行:"+s);
            Thread.Sleep(5000);
        }
    }
}

需要注意的地方,代码中都有注明了,程序运行结果如下:

第二种方法:WaitOne。同样属于“等待”类。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading;

namespace 异步调用实现方法汇总
{
    /// 
    /// 异步调用方法总结:
    /// 1.BeginEnvoke EndEnvoke
    /// 当使用BeginInvoke异步调用方法时,如果方法未执行完,EndInvoke方法就会一直阻塞,直到被调用的方法执行完毕
    /// 
    class Program
    {
        public delegate void PrintDelegate(string s);
        static void Main(string[] args)
        {
            PrintDelegate printDelegate = Print;
            Console.WriteLine("主线程");

            IAsyncResult result= printDelegate.BeginInvoke("Hello World.", null, null);
            Console.WriteLine("主线程继续执行...");
            //当使用BeginInvoke异步调用方法时,如果方法未执行完,EndInvoke方法就会一直阻塞,直到被调用的方法执行完毕
            printDelegate.EndInvoke(result);

            Console.WriteLine("Press any key to continue...");
            Console.ReadKey(true);
        }

        public static void Print(string s)
        {
            Console.WriteLine("异步线程开始执行:"+s);
            Thread.Sleep(5000);
        }
    }
}

第三种方法:轮询。也是属于“等待”类。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading;

namespace 异步调用实现方法汇总3
{
    /// 
    /// 异步调用方法总结:
    /// 3.轮询
    /// 之前提到的两种方法,只能等下异步方法执行完毕,
    /// 在完毕之前没有任何提示信息,整个程序就像没有响应一样,用户体验不好,
    /// 可以通过检查IasyncResult类型的IsCompleted属性来检查异步调用是否完成,
    /// 如果没有完成,则可以适时地显示一些提示信息
    /// 
    class Program
    {
        public delegate void PrintDelegate(string s);
        static void Main(string[] args)
        {
            PrintDelegate printDelegate = Print;
            Console.WriteLine("主线程:"+Thread.CurrentThread.ManagedThreadId );
            IAsyncResult result = printDelegate.BeginInvoke("Hello world.", null, null);
            Console.WriteLine("主线程:" + Thread.CurrentThread.ManagedThreadId + ",继续执行...");
            while (!result.IsCompleted)
            {
                Console.WriteLine(".");
                Thread.Sleep(500);
            }

            Console.WriteLine("主线程:" + Thread.CurrentThread.ManagedThreadId + "  Press any key to continue...");
            Console.ReadKey(true);
        }
        public static void Print(string s)
        {
            Console.WriteLine("当前线程:" + Thread.CurrentThread.ManagedThreadId + s);
            Thread.Sleep(5000);
        }
    }
}

需要注意的地方,代码中都有注明了,程序运行结果如下:

第四种方法:回调。当然属于“回调”类。推荐!!!!

  之前三种方法者在等待异步方法执行完毕后才能拿到执行的结果,期间主线程均处于等待状态。回调和它们最大的区别是,在调用BeginInvoke时只要提供了回调方法,那么主线程就不必要再等待异步线程工作完毕,异步线程在工作结束后会主动调用我们提供的回调方法,并在回调方法中做相应的处理。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading;

namespace 异步调用实现方法汇总4
{
    /// 
    /// 异步调用方法总结:
    /// 4.回调
    /// 之前三种方法者在等待异步方法执行完毕后才能拿到执行的结果,期间主线程均处于等待状态。
    /// 回调和它们最大的区别是,在调用BeginInvoke时只要提供了回调方法,那么主线程就不必要再等待异步线程工作完毕,
    /// 异步线程在工作结束后会主动调用我们提供的回调方法,并在回调方法中做相应的处理,例如显示异步调用的结果。
    /// 
    class Program
    {
        public delegate void PrintDelegate(string s);
        static void Main(string[] args)
        {
            PrintDelegate printDelegate = Print;
            Console.WriteLine("主线程.");
            printDelegate.BeginInvoke("Hello world.", PrintComeplete, printDelegate);
            Console.WriteLine("主线程继续执行...");

            Console.WriteLine("Press any key to continue...");
            Console.ReadKey(true);
        }
        public static void Print(string s)
        { 
            Console.WriteLine("当前线程:"+s);
            Thread.Sleep(5000);
        }
        //回调方法要求
        //1.返回类型为void
        //2.只有一个参数IAsyncResult
        public static void PrintComeplete(IAsyncResult result)
        {
            (result.AsyncState as PrintDelegate).EndInvoke(result);
            Console.WriteLine("当前线程结束." + result.AsyncState.ToString());
        }
    }
}

需要注意的地方,代码中都有注明了,程序运行结果如下:

通过EndInvoke方法得到同步函数的返回值。上面的同步方法返回值为void,我们给个例子:

using System.Diagnostics;
using System.Threading;
using System.Windows;

namespace TestDelegateWrapper
{
    /// 
    /// Interaction logic for MainWindow.xaml
    /// 
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();
        }

        private void ButtonBase_OnClick(object sender, RoutedEventArgs e)
        {
            WrapperSyncMethodAsync("ABC");

            Trace.WriteLine("Main thread continue...");
        }

        private delegate string SyncMethod1Delegate(string str);
        
        private void WrapperSyncMethodAsync(string str)
        {
            SyncMethod1Delegate syncMethod1Delegate = SyncMethod1;
            syncMethod1Delegate.BeginInvoke(str, x =>
            {
                var result= syncMethod1Delegate.EndInvoke(x);

                // using the result to do something
                Trace.WriteLine(result);
            }, null);
        }

        private string SyncMethod1(string str)
        {
            Thread.Sleep(2000);
            return str;
        }
    }
}

输出如下:

Main thread continue...
ABC

以上就是四种实现异步调用函数的四种方法,说的很清楚了,就写这么多~

reference:.NET Framework Delegates: Understanding Asynchronous Delegates

DebugLZQ后续之作:从C#5.0说起:再次总结C#异步调用方法发展史

大家都不是牛人,多多学习,多多交流【请点击下面的“绿色通道”---“关注DebugLZQ”,共同交流进步~】

分类: Design Patterns, Multithreading Asynchronous, .NET Miscellaneous

好文要顶 关注我 收藏该文

相关文章

Java 和.NET,谁将“统治”未来(java和netbeans)

随着时代的变化,技术在不断发展,编程语言的重要性也在发生变化。所以,要我们在两门语言中二选一,通常很困难。在这篇文章里,我们要讨论的是 Java 和.NET 之间的“战争”。.NET 和 Java 是...

.NET Core/.NET5/.NET6 开源项目:工作流组件

前言开源项目是众多组织与个人分享的组件或项目,作者付出的心血我们是无法体会的,所以首先大家要心存感激、尊重。请严格遵守每个项目的开源协议后再使用。尊重知识产权,共建和谐开源社区。ELSAElsa Co...

谁说.NET没有GC调优,只改一行代码就让程序不再占用内存

经常看到有群友调侃“为什么搞Java的总在学习JVM调优?那是因为Java烂!我们.NET就不需要搞这些!”真的是这样吗?今天我就用一个案例来分析一下。昨天,一位学生问了我一个问题:他建了一个默认的A...

java中的url 编码与解码(java urldecode解码)

在开始讨论编码解码之前,首先来明确一下问题。什么是application/x-www-form-urlencoded字符串?答:它是一种编码类型。当URL地址里包含非西欧字符的字符串时,系统会将这些字...

.NET程序员学习JAVA的最佳开源项目RuoYi

随.NET生态的快速发展,以及跨平台产品.NET Core的出现,.NET的开源生态也变得越来越繁荣,.NET程序员终于可以扬眉吐气的说.NET跨平台了!基于.NET平台,可以快速开发Web程序、桌面...

工作日报 2021.10.20 OkHttp3错误异常:unexpected end of stream

2021.10.20问题澄清:Plugtest IDMS对接版本准备;蓝牙手咪适配问题配合定位;Mcdata http彩信上传,接口方案DT联调;河北联创项目联调;OkHttp3错误异常: java....