BT

如何利用碎片时间提升技术认知与能力? 点击获取答案

Silverlight的多线程能力(上)

| 作者 吴磊 关注 0 他的粉丝 发布于 2011年3月10日. 估计阅读时间: 11 分钟 | QCon上海2018 关注大数据平台技术选型、搭建、系统迁移和优化的经验。

对于多线程其实一直以来都存在很多误区:比如多任务与多线程就很容易被混为一谈,而多线程也常被理所应当的认为是并行等等。而事实却是:多任务≠多线程、单任务≠单线程、多线程不一定并行,多线程与性能不成线性关系等等,其中道理在这里不再详述。笔者认为Silverlight多线程主要作用不是在于提高性能,而是在于用户体验,其根本目的是解决用户体验中的响应速度,减少单线程带来的阻塞问题。用一个贴切的例子来形容单线程和多线程的区别:单线程就好像只有一个服务窗口卖票的车站,人们排队买票时都是单线程处理的,而且不能抢夺位置,这样只要前方有一个人出现长时间等待,后面的人都不能被响应,这就出现了单线程阻塞;而多线程就好像有多个服务窗口去卖票,这样车票买卖和等待的情况就会好很多(当然这个例子如果换成公共厕所,对于用户体验就显得更为重要了)。

这次我们就要来看看Silverlight的多线程能力,其实Silverlight的多线程体现在两大方面:

第一方面是将UI线程与后台工作线程的分离,使得UI线程可以更好地响应用户操作,而后台线程处理完后,允许通过异步的方式将处理结果推回前台进行展示。笔者认为这是多线程在Silverlight中最主要的作用(很多传统Web应用开发者在刚开始接触Silverlight时很不适应这种前后台线程的异步操作)。

第二方面是对后台作业的多线程支持,比如当需要在客户端后台并行运算时,你可以通过发起多个线程来完成这些运算。在上期《Silverlight CoreCLR结构浅析》中,我已经给大家介绍了Silverlight的基础类库,其中就包括多线程的相关类集。

UI线程是Silverlight与用户交互的线程,在Silverlight中UI线程是单一的,其中装入的是UI控件类及用于数据绑定的View Model类(什么是View Model?就是只为View层服务的实体,如果要展开说会很长,就此打住!),而在后台线程中是不能直接访问这些UI线程中的数据与控件对象的属性。但大家不用担心,Silverlight和WPF的线程模型都使用了类似于Java Swing中EDT(Event Dispatch Thread)这种安全的事件分发线程模型来解决UI线程与其他后台线程的数据互访问题。在Silverlight(WPF)的控件类库System.Windows下所有类都继承了DependencyObject基类,DependencyObject类不仅提供了Silverlight(WPF)最基础的依赖性属性服务(什么是依赖性属性?简单的说就是对象属性值依赖于其他计算值的方式,这种方式为数据绑定、动画、重用样式都提供了可行性,这里不再展开),同时也开启了UI线程与后台线程的数据互访通道,在DependencyObject中有一个非常重要的属性——Dispatcher,后台线程可以通过调用发起者(一般都是UI控件)的Dispatcher来实现互操作,后台线程可以通过下面的方式来直接操作UI线程中的对象:

_UISender.Dispatcher.BeginInvoke(() => 
{ 
    //这里可以访问UI线程中的对象,因为这个委托本身就在UI线程中执行 
}

上面的()=>是Lamda表达式中对于无入参的委托方法的简写形式,如果有传入参数可以在括号中列明,当然你也可以使用Action各种重载到其他地方实现委托过程。

如果要实现UI线程创建并访问后台线程就更加简单,Silverlight提供了多种创建后台线程的方式:

  • 基于普通的System.Threading.Thread类创建后台线程
  • Thread类是最基础的多线程类,它可以创建一个独立运行的线程,比如:

    Threadthread = new Thread(obj.functionName); 
    thread.IsBackground = true; 
    thread.Start();

    但Thread对于线程的监控、销毁、回调都比较复杂,因此笔者往往使用Thread来完成一些简单的且不需要回调的任务。

  • 基于System.Windows.Threading.DispatchTimer类创建后台定时器线程
  • DispatchTimer类是Silverlight(WPF)里才出现的后台线程定时器,相较于原有System.Threading.Timer差别在于DispatchTimer是真正的在后台线程内独立执行,而Timer仍然在UI线程中执行,只是定时获得UI线程控制权而已。DispatchTimer只适合于定时执行的任务,你可以根据需要来设置等待时歇,其创建方式如下:

    DispatcherTimer dt = new DispatcherTimer(); 
    dt.Interval = new TimeSpan(0, 0, 0, 0, 10); 
    dt.Tick += new EventHandler(dt_Tick); 
    dt.Start(); 
    void dt_Tick(object sender, EventArgs e) 
    { 
        //超过等待时歇时发生 
        //这里可以访问UI线程中的对象 
        //如果要结束定时器可以调用dt.Stop(); 
    }

    DispatcherTimer其实也是除StoryBoard外可以实现动画的重要组件,当然要慎用DispatcherTimer来构建过多后台线程,否则会使CPU调度开销增加反而影响效率!(调度开销将在下部分讲解)

  • 基于System.ComponentModel.BackgroundWorker类轻松创建后台线程
  • 微软在WinForm架构中就引入了BackgroundWorker类,这个类内建了许多线程包装方法,从而大大简化线程交互的编码过程。在Silverlight(WPF)中也可以通过BackgroundWorker类来轻松创建后台线程,其创建方式如下:

    BackgroundWorker bw = new BackgroundWorker(); 
    bw.DoWork += new DoWorkEventHandler((object, doworkeventarg) =>obj.function()); 
    bw.RunWorkerCompleted += new RunWorkerCompletedEventHandler(bw_RunWorkerCompleted); 
    if(!bw.IsBusy) bw.RunWorkerAsync(); 
    void bw_RunWorkerCompleted(object sender, RunWorkerCompletedEventArgs e) 
    { 
        //任务完成时回调事件 
    }

    BackgroundWorker也可以通过ReportProgress(intpercentProgress)方法来向其他线程报告其进度完成情况,当然这只适合于可量化进度的后台工作线程,其实现如下:

    //后台线程obj.function()中 
    … 
    obj.ReportProgress(i); 
    … 
    //在UI线程中定义报告时间委托 
    bw.ProgressChanged += new ProgressChangedEventHandler(bw_ProgressChanged); 
    void bw_ProgressChanged(object sender, ProgressChangedEventArgs e) 
    { 
        //显示进度的相关处理方法 
    }
  • 基于System.Threading.ThreadPool静态类创建后台线程
  • 在所有多线程解决方案中,ThreadPool线程池是笔者最常用的技术。其优点在于易于控制并且减少开销,在线程池中的线程不会由于完成一个任务就消亡,而是会继续执行其他的任务,大大减少了线程的创建与销毁开销。ThreadPool中的QueueUserWorkItem方法可以将任何处理函数排入后台线程队列中执行,其创建方式也非常简单:

    obj.OnEvent += (object, eventarg) => Dispatcher.BeginInvoke(UI_OnEvent); 
    ThreadPool.QueueUserWorkItem(state =>obj.function(), stat); 
    voidUI_OnEvent() 
    { 
        //后台线程事件发生时的回调事件 
    }

    在后台线程对象obj中你可以随意定义回调事件,并通过Dispatcher.BeginInvoke的方法来通知UI线程的委托。这样的方式比较简单而且实用。当然ThreadPool类还提供了RegisterWaitForSingleObject方法来实现Timer定时器的功能,可以说ThreadPool是在企业应用中比较常用的多线程实现类。

至此,就给大家介绍了Silverlight常用多线程实现方式,但笔者还要强调的是:Silverlight的多线程是为了提升用户体验。其实Silverlight开发围绕的关键是用户体验,用户体验在现代商业应用开发中的地位非常的重要。本主题的下半部分,笔者将通过一个实例来为大家讲述Silverlight的多线程性能,以及与其他Web开发技术的性能对比。

评价本文

专业度
风格

您好,朋友!

您需要 注册一个InfoQ账号 或者 才能进行评论。在您完成注册后还需要进行一些设置。

获得来自InfoQ的更多体验。

告诉我们您的想法

允许的HTML标签: a,b,br,blockquote,i,li,pre,u,ul,p

当有人回复此评论时请E-mail通知我

深刻 by lee xiao

分析的很到位!很清楚!

指正一处错误 by lian peixun

System.Threading.DispatcherTimer应该为System.Windows.Threading.DispatcherTimer.

Re: 指正一处错误 by 吴 磊

多谢提醒,已经更正

学习了 by 王 建平

在银光中国看到过同一篇文章

Re: 学习了 by 吴 磊

是我后来转过去的

允许的HTML标签: a,b,br,blockquote,i,li,pre,u,ul,p

当有人回复此评论时请E-mail通知我

允许的HTML标签: a,b,br,blockquote,i,li,pre,u,ul,p

当有人回复此评论时请E-mail通知我

5 讨论

登陆InfoQ,与你最关心的话题互动。


找回密码....

Follow

关注你最喜爱的话题和作者

快速浏览网站内你所感兴趣话题的精选内容。

Like

内容自由定制

选择想要阅读的主题和喜爱的作者定制自己的新闻源。

Notifications

获取更新

设置通知机制以获取内容更新对您而言是否重要

BT