Reactive Extensions(Rx)并发浅析
Reactive Extensions(Rx)并发浅析
iSun
Design & Code
.Net并行编程 - Reactive Extensions(Rx)并发浅析
关于Reactive Extensions(Rx)
关于Reactive Extensions(Rx),先来看一下来自微软的官方描述:
The Reactive Extensions (Rx) is a library for composing asynchronous and event-based programs using observable sequences and LINQ-style query operators. Using Rx, developers represent asynchronous data streams with Observables, query asynchronous data streams using LINQ operators, and parameterize the concurrency in the asynchronous data streams using Schedulers. Simply put, Rx = Observables + LINQ + Schedulers.
“Reactive Extensions(Rx)是一个类库,它集成了异步、基于可观察(observable)序列的事件驱动编程和LINQ-style的查询操作。使用Rx,开发人员可以用observable对象描述异步数据流,使用LINQ操作符异步查询数据和使用Schedulers控制异步过程中的并发。简而言之,Rx = Observables + LINQ + Schedulers。”
Reactive Extensions(Rx)就一定是多线程?
在以上的描述中,反复出现了一个词“异步”。一般来讲,提到“异步”,首先反应到的就是多线程。那问题来了,使用Reactive Extensions就一定意味着多线程吗?先来看一个示例,代码来了:

1 static void Main(string[] args)
2 {
3 Console.WriteLine("Starting on threadId:{0}", Thread.CurrentThread.ManagedThreadId);
4 var sub = new Subject<Object>();
5
6 sub.Subscribe(o => Console.WriteLine("Received {1} on threadId:{0}", //为Observable订阅处理器(handler),输出handler thread id
7 Thread.CurrentThread.ManagedThreadId,
8 o));
9 ParameterizedThreadStart notify = obj => //委托定义,其内输出被观察对象的thread id
10 {
11 Console.WriteLine("OnNext({1}) on threadId:{0}",
12 Thread.CurrentThread.ManagedThreadId,
13 obj);
14 sub.OnNext(obj);
15 };
16 notify(1);
17 new Thread(notify).Start(2);
18 new Thread(notify).Start(3);
19
20 Console.Read();
21 }

代码中,分别输出了通知者的thread id和callback handler的thread id。这里使用的是Rx默认的线程并发方式。输出结果如下:
无论是在当前线程调用,还是新启线程执行,通知者和处理方法所在线程均为同一个。在该示例中,Rx的线程分配是在free-threaded模式下工作的,free-threaded就意味着我们不强行指其Rx中的subscription, notification执行线程。这是Rx的默认工作模式,而这种模式下subscribing/call OnNext并没有引发新的线程来处理observable序列,线处理方式是单线程(Single Threaded Apartment,STA)。所以,我们可以这样说:单线程是Reactive Extensions(Rx)的默认处理方式。
使用SubscribeOn控制订阅(subscribing)的上下文
IObservable<TSource>的扩展方法SubscribeOn<TSource>(IScheduler)允许我们传入一调度器(Scheduler),控制订阅执行的上下文。

1 static void Main(string[] args)
2 {
3 Console.WriteLine("Starting on threadId:{0}", Thread.CurrentThread.ManagedThreadId);
4 var source = Observable.Create<int>(
5 o =>
6 {
7 Console.WriteLine("Invoked on threadId:{0}", Thread.CurrentThread.ManagedThreadId);
8 o.OnNext(1);
9 o.OnNext(2);
10 o.OnNext(3);
11 o.OnCompleted();
12 Console.WriteLine("Finished on threadId:{0}",
13 Thread.CurrentThread.ManagedThreadId);
14 return Disposable.Empty;
15 });
16 source
17 //.SubscribeOn(Scheduler.ThreadPool)
18 .Subscribe(
19 o => Console.WriteLine("Received {1} on threadId:{0}",
20 Thread.CurrentThread.ManagedThreadId,
21 o),
22 () => Console.WriteLine("OnCompleted on threadId:{0}",
23 Thread.CurrentThread.ManagedThreadId));
24 Console.WriteLine("Subscribed on threadId:{0}", Thread.CurrentThread.ManagedThreadId);
25
26 Console.Read();
27 }

代码中,使用Observable.Create创建一Observable序列,随后订阅该序列。输出结果为:
当序列被订阅source.Subscribe,代理Observable.Create被调用执行。首先是OnNext(1) handler,依次是OnNext(2) OnNext(3) handler和OnCompleted handler,最后执行到“Subscribed on threadId:10”。整个过程是线性的,阻塞(block)式的。这是符合上面分析的Rx默认单线程的模式的。
我们放开.SubscribeOn(Scheduler.ThreadPool)的注释,指定Rx工作在线程池内完成。执行结果如下:
可以看到,所有的handler都是在一新线程内完成的。这是一个非阻塞的(no=-block)模式。
SubscribeOn方法常用来指定Observable notifications的线程执行模式(哪里执行)。其常用于以下的场景中:
》 UI线程不允许阻塞
》 不需在UI线程中更新显示
常用的Scheduler属性:
CurrentThread | 在当前进程中尽快的调度工作,同步(synchronous,block) |
Immediate | 在当前进程中立即调度工作,同步(synchronous,block) |
NewThread | 在新线程中调度工作(asynchronous,no-block) |
TaskPool | 在任务工厂中调度工作(asynchronous,no-block) |
ThreadPool | 在线程池中调度工作(asynchronous,no-block) |
参考资料:
Reactive Extensions(Rx)并发浅析的更多相关文章
- Reactive Extensions(Rx) 学习
Bruce Eckel(著有多部编程书籍)和Jonas Boner(Akka的缔造者和Typesafe的CTO)发表了“反应性宣言”,在其中尝试着定义什么是反应性应用. 这样的应用应该能够: 对事件做 ...
- Reactive Extensions (Rx) 入门(5) —— Rx的事件编程
译文:https://blog.csdn.net/fangxing80/article/details/7749907 原文:http://www.atmarkit.co.jp/fdotnet/int ...
- Reactive Extensions (Rx) 入门(4) —— Rx的事件编程
译文:https://blog.csdn.net/fangxing80/article/details/7685393 原文:http://www.atmarkit.co.jp/fdotnet/int ...
- Reactive Extensions (Rx) 入门(3) —— Rx的事件编程
译文:https://blog.csdn.net/fangxing80/article/details/7628322 原文:http://www.atmarkit.co.jp/fdotnet/int ...
- Reactive Extensions (Rx) 入门(2) —— 安装 Reactive Extensions
译文:https://blog.csdn.net/fangxing80/article/details/7581937 原文:http://www.atmarkit.co.jp/fdotnet/int ...
- Reactive Extensions (Rx) 入门(1) —— Reactive Extensions 概要
译文:https://blog.csdn.net/fangxing80/article/details/7381619 原文:http://www.atmarkit.co.jp/fdotnet/int ...
- .Net并行编程 - Reactive Extensions(Rx)并发浅析
关于Reactive Extensions(Rx) 关于Reactive Extensions(Rx),先来看一下来自微软的官方描述: The Reactive Extensions (Rx) is ...
- Rx (Reactive Extensions)介绍
Reactive Extensions (Rx) 原来是由微软提出的一个综合了异步和基于事件驱动编程的库包,使用可观察序列和LINQ-style查询操作. 使用Rx, 开发者可以用Observable ...
- Rx (Reactive Extensions)
The Reactive Extensions (Rx) is a library for composing asynchronous and event-based programs using ...
随机推荐
- telnet发电子邮件
无聊今天的工作,想想一个学生被提到最后一次telnet发电子邮件,所以我想试试.最后,成功的实践,这里做个总结. 首先,cmd进telnet打开回话: 下面红色字体为命令. 1.open smtp.1 ...
- 迅雷云加速开放平台c#demo
迅雷云加速开放平台c#demo.很多人很遇到下载文件的问题.这个例子是调用迅雷云加速开放平台的dll,进行下载,速度很快,下载过程中可以获取到很全的下载信息,比如下载速度,进度,完成状态等. 例子中带 ...
- dlmalloc 2.8.6 源代码具体解释(6)
本文章由vector03原创, 转载请注明出处. 邮箱地址: mmzsmm@163.com, 欢迎来信讨论. 3.4 sys_alloc sys_alloc是dlmalloc中向系统获取内存的主要接口 ...
- C/S通信模型和相关技术要点
差点儿全部的项目中,都会涉及到client和服务端.而client与server之间的通信又是一个非经常见但又有须要问题的技术问题. 首先,连接方式有长连接和短连接.先看看概念. 长连接短连接仅仅是一 ...
- 【原创】构建高性能ASP.NET站点之一 剖析页面的处理过程(前端)
原文:[原创]构建高性能ASP.NET站点之一 剖析页面的处理过程(前端) 构建高性能ASP.NET站点之一 剖析页面的处理过程(前端) 前言:在对ASP.NET网站进行优化的时候,往往不是只是懂得A ...
- 关于“类不能支持Automation操作”错误的解决方法
一段程序IE上老是提示“类不支持Automation操作”的错误,IE6.7.8都一样,但是Firefox可以,后来网上找到如下解决方法: 重新注册下以下文件,问题便解决了:msscript.ocxd ...
- Data source rejected establishment of connection, message from server: "Too many connections"
错误叙述性说明: 測试一段时间没有不论什么问题.今天突然用户无法登录,报错如Data source rejected establishment of connection, message fro ...
- 使用EasyUI实现加入和删除功能
增删该查是不论什么一个项目都少不了的功能操作.这篇博文主要简单介绍一下怎样使用EasyUI实现加入和删除功能. 首先.导入EasyUI的js代码: <link href="~/Easy ...
- Java里泛型有什么作用
1 泛型赋予了类型參数式多态的能力 2 泛型的第一个优点是编译时的严格类型检查,提高了程序的安全性和健壮性,这是集合框架最重要的特点. 3 泛型消除了绝大多数的类型转换.假设没有泛型,当你使用集合框架 ...
- Cts框架解析(7)-任务运行的调度室
TestInvocation /** * {@inheritDoc} */ @Override public void invoke(ITestDevice device, IConfiguratio ...