C#沉淀-异步编程 四

BackgroundWorker类

async/await特性适合那些需要在后台完成的不相关的小任务,有时候,需要另建一个线程,在后台持续运行以完成某项工作,并不时地与主线程进行通信,BackgroundWorker类就是为此而生。

BackgroundWorker类的主要成员

  • 属性
    • WorkerReportsProgress 设置后台任务是否可以把它的进度汇报给主线程
    • WorkerSupportsCancellation 是否支持从主线程取消
    • IsBusy 检查后台任务是否正在运行
    • CancelationPending 状态值,布尔类型
  • 方法
    • RunWorkerAsync() 获取后台线程并执行DoWork事件处理程序
    • CancelAsync() 把CancellationPending属性设置为true。DoWork事件处理程序需要检查这个属性来决定是否应该停止处理
    • ReportProgress() DoWork工作的时候向主线程汇报进度
  • 事件
    • DoWork 在后台线程开始的时候触发
    • ProgressChanged 在后台任务汇报状态的时候触发
    • RunWorkerCompleted 在后台工作线程退出的时候触发

使用示例:

using System.Threading;
using System.Windows;
using System.ComponentModel;

namespace WpfForAsync
{
    /// <summary>
    /// MainWindow.xaml 的交互逻辑
    /// </summary>
    public partial class MainWindow : Window
    {
        // 首先需要创建BackgroundWorker对象
        BackgroundWorker bgWorker = new BackgroundWorker();

        public MainWindow()
        {
            InitializeComponent();

            // 设置BackgroundWorker属性
            // WorkerReportsProgress设置工作线程为主线程汇报进度
            bgWorker.WorkerReportsProgress = true;
            // 设置支持从主线程取消工作线程
            bgWorker.WorkerSupportsCancellation = true;

            // 连接BackgroundWorkder对象的处理程序
            //DoWork是必须的
            bgWorker.DoWork += BgWorker_DoWork;
            //ProgressChanged与RunWorkerCompleted这两个是可选的
            bgWorker.ProgressChanged += BgWorker_ProgressChanged;
            bgWorker.RunWorkerCompleted += BgWorker_RunWorkerCompleted;
        }

        private void BgWorker_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e)
        {
            progressBar.Value = 0;

            if (e.Cancelled)
            {
                MessageBox.Show("Process was cancelled.", "Process Cancelled");
            }
            else
            {
                MessageBox.Show("Process completed normally.", "Process Completed");
            }
        }

        private void BgWorker_ProgressChanged(object sender, ProgressChangedEventArgs e)
        {
            progressBar.Value = e.ProgressPercentage;
        }

        /// <summary>
        /// 这里是希望在后台工作的代码 
        /// 它是在独立的线程中执行的
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void BgWorker_DoWork(object sender, DoWorkEventArgs e)
        {
            BackgroundWorker worker = sender as BackgroundWorker;

            for (int i = 0; i <= 10; i++)
            {
                if (worker.CancellationPending)
                {
                    e.Cancel = true;
                    break;
                }
                else
                {
                    /// 后台线程通过调用ReportProgress方法与主线程通信。
                    /// 届时将触发ProgressChanged事件
                    /// 主线程可以处理附加到ProgressChanged事件上的处理程序
                    worker.ReportProgress(i * 10);
                    Thread.Sleep(500);
                }
            }
        }

        private void btnProcess_Click(object sender, RoutedEventArgs e)
        {
            if (!bgWorker.IsBusy)
            {
                //主线程调用RunWorkerAsync的时候,会触发DoWork事件
                bgWorker.RunWorkerAsync();
            }
        }

        private void progressBar_Click(object sender, RoutedEventArgs e)
        {
            // 通知后台线程退出
            // 此时会将后台线程类的CancellationPending属性设置为true
            // DoWork事件处理程序会定期检查CancellationPending确定是否要退出
            // 如果后台线程退出,则会触发RunWorkerCompleted事件
            bgWorker.CancelAsync();
        }
    }
}

这是一个WPF程序,前端代码与上一节的一样,后端代码使用后台线程处理

这里再来谈一下每个事件处理程序形参中的参数类EventArgs的结构(所包含的属性)

  • DoWorkEventArgs
    • Argument
    • Result
    • Cancel
  • ProgressChangedEventArgs
    • ProgressPercentage
    • UserState
  • RunWorkerCompletedEventArgs
    • Cancelled
    • Error
    • Result
    • UserState

并行循环

这里介绍Parallel.For循环和Parallel.ForEach循环,它们都们于System.Threading.Tasks命名空间下

这两个循环可以使迭代之间彼此独立,并且程序运行在多核处理器的机器上时,能将不同的迭代放在不同的处理器在上并行处理

Parallel.For方法有很多个重载,这里介绍其中一个

public static ParallelLoopResult For(int fromInclusive, int toExclusive, Action<int> body);
  • fromInclusive参数是迭代系列的第一个整数
  • toExclusive参数是比迭代系列最后一个索引号大1的整数
  • body是接受单个输入参数的委托,body的代码在每一次迭代中执行一次

示例:

using System;
using System.Threading.Tasks;  // 必须使用这个命名空间

namespace CodeForAsync
{
    class Program
    {
        static void Main(string[] args)
        {
            // 从0开始迭代,直到14
            // 打印出索引与索引的平方
            Parallel.For(0, 15, i => 
                         Console.WriteLine("Thre Square of {0} is {1}", i, i * i));
            Console.ReadKey();
        }
    }
}

输出:

Thre Square of 0 is 0
Thre Square of 2 is 4
Thre Square of 1 is 1
Thre Square of 4 is 16
Thre Square of 6 is 36
Thre Square of 7 is 49
Thre Square of 9 is 81
Thre Square of 10 is 100
Thre Square of 11 is 121
Thre Square of 12 is 144
Thre Square of 13 is 169
Thre Square of 8 is 64
Thre Square of 14 is 196
Thre Square of 3 is 9
Thre Square of 5 is 25

另一个示例:

using System;
using System.Threading.Tasks;

namespace CodeForAsync
{

    class Program
    {

        static void Main(string[] args)
        {
            const int maxValues = 50;
            int[] squares = new int[maxValues];

            Parallel.For(0, maxValues, i => squares[i] = i * i);

            for (int i = 0; i < squares.Length; i++)
            {
                Console.WriteLine(squares[i]);
            }

            Console.ReadKey();
        }
    }
}

本便将索引的平方添加到一个数组中,虽然是并行操作,但是数组里的值却是按顺序按序的

Parallel.ForEach示例

Parallel.ForEach同样有许多重载,这里介绍一个简单的

public static ParallelLoopResult ForEach<TSource>(IEnumerable<TSource> source, Action<TSource> body);
  • source是被操作的数据源
  • body是一个无返回值的委托
using System;
using System.Threading.Tasks;

namespace CodeForAsync
{

    class Program
    {

        static void Main(string[] args)
        {
            string[] squares = new string[] { "We", "hold", "these", "truths", "to", "be", "self-evident", "that", "all", "men", "are", "create", "equal" };

            Parallel.ForEach(squares, i => Console.WriteLine(string.Format("{0} has {1} letters", i, i.Length)));

            Console.ReadKey();
        }
    }
}

输出:

hold has 4 letters
We has 2 letters
to has 2 letters
be has 2 letters
self-evident has 12 letters
all has 3 letters
men has 3 letters
are has 3 letters
create has 6 letters
equal has 5 letters
these has 5 letters
truths has 6 letters
that has 4 letters

可以看出,这两个循环在操作的时候,每次迭代所输出的顺序都不一样,因为他们是并行循环,也就是将多个迭代放在多核上处理

其他异步编程模式

当委托对象被调用时,它调用了它的调用列表中包含的方法

如果委托对象在调用列表中只有一个方法(引用方法),它就可以异步执行这个方法。委托类有两个方法,叫做BeginInvoke和EndInvoke,它们就是用来这么做的

  • 当调用委托的BeginInvoke方法,它开始在一个独立线程上执行引用方法,并且立即返回到原始线程。原始线程可以继续,而引用方法会在线程池的线程中并行执行。
  • 当程序希望获取已完成的异步方法的结果时,可以检查BeginInvoke返回的IAsyncResult的IsCompleted属性,或调用委托的EndInvoke该来 等等委托完成
  • 等等——完成模式中,在发起了异步方法以及做了一些其他处理之后,原始线程就中断并且等异步方法完成之后再继续
  • 轮询模式中,原始线程定期检查发起的线和是否完成,如果没有则可以继续做一些其他的事情
  • 回调模式中,原始线程一直执行,无需等等或检查发起的线程是否完成。在发起的线和中的引用方法完成之后,发起的线程就会调用回调方法,由回调方法在调用EndInvoke之前处理异步方法的结果

BeginInvoke和EndInvlke

BeginInvoke的使用

  • 在使用BeginInvoke时,参数列表中的实际参数组成如下:
    • 引用方法需要的参数
    • 两个额外的参数——callbak参数和state参数
  • BeginInvoke从线程池中获取一个线程并且让引用方法在新的线程中开始运行
  • BeginInvoke返回给调用线程一个实现IAsyncResult接口的对象的引用。这个接口引用包含了在线程池线程中运行的异步方法的当前状态,原始线程然后可以继续执行

示例:

using System;
using System.Threading.Tasks;

namespace CodeForAsync
{
    delegate long MyDel(int first, int second);

    class Program
    {
        static long Sum(int x, int y)
        {
            int z = x + y;
            return (long)z;
        }
        static void Main(string[] args)
        {
            MyDel del = new MyDel(Sum);
            IAsyncResult iar = del.BeginInvoke(3, 5, null, null);
            // 3, 5是引用方法所需要的参数
            // null, null分别表示回调方法与状态没有值
            // 此时从线程池获取一个一线程,并且在新的线程上运行Sum方法
            // 收集新线程的状态返回给调用者

            Console.ReadKey();
        }
    }
}

EndInvoke的使用

EndInvlke方法用来获取由异步方法调用返回的值,并且线程使用的资源,其特性如下:

  • 它接受一个由BeginInvoke方法返回的IAsyncResult对象的引用,并找到它关联的线程
  • 如果线程池的线程已经退出,EndInvlke做如下的事情
    • 它清理退出线程的状态并且释放其资源
    • 它找到引用方法返回的值并且把它作为返回值
  • 如果当EndInvlke被调用时线程池的线程仍然在运行,调用线程就会停止并等等,直到清理完毕并返回值。因为EndInvlke是为开户的线程进行清理,所以必须确保对每一个BeginInvlke都调用EndInvlke
  • 如果异步方法触发了异步,在调用EndInvlke时会抛出异常

示例:

using System;
using System.Threading.Tasks;

namespace CodeForAsync
{
    delegate long MyDel(int first, int second);

    class Program
    {
        static long Sum(int x, int y)
        {
            int z = x + y;
            return (long)z;
        }
        static void Main(string[] args)
        {
            MyDel del = new MyDel(Sum);
            IAsyncResult iar = del.BeginInvoke(3, 5, null, null);

            //必须将IAsyncResult对象作为参数
            //如果委托引用的方法有ref和out参数,也必须包含在EndInvoke参数列表中
            long result = del.EndInvoke(iar);

            Console.ReadKey();
        }
    }
}

来看一个完整的示例:

using System;
using System.Threading;
using System.Threading.Tasks;

namespace CodeForAsync
{
    delegate long MyDel(int first, int second);

    class Program
    {
        static long Sum(int x, int y)
        {
            Console.WriteLine("                   Inside Sum");
            Thread.Sleep(100);

            return x + y;
        }
        static void Main(string[] args)
        {
            MyDel del = new MyDel(Sum);

            Console.WriteLine("Before BeginInvoke");
            IAsyncResult iar = del.BeginInvoke(3, 5, null, null);  // 开始异步调用 
            Console.WriteLine("After BeginInvoke");

            Console.WriteLine("Ding Stuff");  // 在这里可以做点其他事情

            long result = del.EndInvoke(iar);  // 等等结束并获取结果
            Console.WriteLine("After EndInvoke: {0}",result);

            Console.ReadKey();
        }
    }
}

输出:

Before BeginInvoke
After BeginInvoke
Ding Stuff
                   Inside Sum
After EndInvoke: 8

AsyncResult类

  • 当调用委托对象的BeginInvoke方法时,系统创建了一个AsyncResult类的对象。然而,它不返回类对象的引用,而是返回对象中包含的IAsyncResult接口的引用
  • AsyncResult对象包含一个叫做AsyncDelegate的属性,它返回一个指向被调用来开户异步方法的委托的引用。但是,这个属性是类对象的一部分而不是接口的一部分
  • 该类还包含一个IsCompleted属性,返回一个布尔值,表示方法是否完成
  • 该类还包含一个AsyncState属性,返回一个对象的引用,作为BeginInvoke方法调用时的state参数。它返回object类型的引用

轮询模式

在轮询模式中,原始线程发起了异步方法的调用,做一些其他处理,然后使用IAsyncResult对象的IsComplete属性来定期检查开户的线程是否完成。如果异步方法已经完成 ,原始线程就调用EenInvoke并继续。否则,它做一些其他处理,然后过一会再检查

示例:

using System;
using System.Threading;
using System.Threading.Tasks;

namespace CodeForAsync
{
    delegate long MyDel(int first, int second);

    class Program
    {
        static long Sum(int x, int y)
        {
            Console.WriteLine("                   Inside Sum");
            Thread.Sleep(100);

            return x + y;
        }
        static void Main(string[] args)
        {
            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");

                //继续处理
                for (int i = 0; i < 10000; i++) ;
            }

            Console.WriteLine("Dnoe");  

            long result = del.EndInvoke(iar);  // 等等结束并获取结果
            Console.WriteLine("Result: {0}",result);

            Console.ReadKey();
        }
    }
}

回调模式

回调模式中,一旦初始线程发起了异步方法,它就只管自己,不再考虑同步。当异步方法调用结束之后,系统调用一个用户自定义的方法来处理结果,并且调用委托的EndInvoke方法。这个用户自定义的方法叫做回调方法回调

语法示例:

IAsyncResult iar = del.BeginInvoke(3, 5, new AsyncCallback(CallWhenDone), null);
// or
IAsyncResult iar = del.BeginInvoke(3, 5, CallWhenDone , null);

从示例中了解真相:

using System;
using System.Runtime.Remoting.Messaging;  // 调用AsyncResult类型
using System.Threading;

namespace CodeForAsync
{
    delegate long MyDel(int first, int second);

    class Program
    {
        static long Sum(int x, int y)
        {
            Console.WriteLine("             Inside Sum");
            Thread.Sleep(100);

            return x + y;
        }

        /// <summary>
        /// 定义一个方法,用作回调
        /// 这里想在回调方法中使用委托对象的EndInvoke来处理异步方法执行后的输出
        /// 但是委托对象在主线程里,而非在异步线程里
        /// 所以,这里将IAsyncResult对象作为参数
        /// </summary>
        /// <param name="iar"></param>
        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("             Thre result is: {0}", result);
        }

        static void Main(string[] args)
        {
            MyDel del = new MyDel(Sum);

            Console.WriteLine("Before BeginInvoke");

            //state参数接收一个object类型对象
            //为了在回调方法中使用委托del,这里将del对象作为状态类传入
            IAsyncResult iar = del.BeginInvoke(3, 5, new AsyncCallback(CallWhenDone), del);  // 开始异步调用 
            Console.WriteLine("Doing more work in Main");
            Thread.Sleep(500);
            Console.WriteLine("Done with Main. Exiting.");

            Console.ReadKey();
        }
    }
}

输出:

Before BeginInvoke
Doing more work in Main
             Inside Sum
             Inside CallWhenDone
             Thre result is: 8
Done with Main. Exiting.

计时器

这里介绍System.Threading命名空间中的那个Timer类

  • 计时器在每次时间到期之后调用回调方法。回调方法必须是TimerCallback委托形式的,结构如下所示。它接受了一个object类型作为参数,并且返回类型是void

    void TimerCallback(object state)
    
  • 相关属性:

    • dueTime:是回调方法首次被调用之前的时间
    • period:是两次成功调用回调函数之间的时间间隔
    • state: 可以是null或在每次回调方法执行时机传入的对象的引用

Timer常用的构造如下:

Timer(TimerCallback callback, object state, uint dueTime, uint period)

示例:

using System;
using System.Runtime.Remoting.Messaging;  // 调用AsyncResult类型
using System.Threading;

namespace CodeForAsync
{
    delegate long MyDel(int first, int second);

    class Program
    {
        int TimesCalled = 0;

        void Display(object state)
        {
            Console.WriteLine("{0} {1}",state.ToString(), ++TimesCalled);
        }

        static void Main(string[] args)
        {
            Program p = new Program();

            //参数分别为:回调方法、状态信息、开始前时间、间隔时间
            Timer myTimer = new Timer(p.Display, "Processing timer event", 2000, 1000);
            Console.WriteLine("Timer started.");

            Console.ReadKey();
        }
    }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,128评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,316评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,737评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,283评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,384评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,458评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,467评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,251评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,688评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,980评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,155评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,818评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,492评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,142评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,382评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,020评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,044评论 2 352

推荐阅读更多精彩内容

  • 1.ios高性能编程 (1).内层 最小的内层平均值和峰值(2).耗电量 高效的算法和数据结构(3).初始化时...
    欧辰_OSR阅读 29,353评论 8 265
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,094评论 1 32
  • OC语言基础 1.类与对象 类方法 OC的类方法只有2种:静态方法和实例方法两种 在OC中,只要方法声明在@int...
    奇异果好补阅读 4,264评论 0 11
  • 幸福是需要修出来的~每天进步1%~幸福实修14班~01组李玉珍# 20180217(26/60) 【幸福实修目标】...
    stx2010阅读 251评论 0 3
  • 三更点将台,击鼓到天明。 骤把晴空取,木秀耳目清。
    风十二郎阅读 244评论 0 3