Rx主题中观察者的数量

时间:2012-11-29 21:06:24

标签: system.reactive

使用Rx,获取主题中当前观察者数量的最佳方法是什么?

我有一个场景,我想发布一条消息,但前提是有观察者。如果没有观察员,我需要做点别的事。

为了解决这个问题,我所做的是创建自己的ISubject实现并公开内部IObserver集合的计数。我确信必须有一个开箱即用的方式,我只是不完全熟悉Rx提供的东西。

谢谢!

3 个答案:

答案 0 :(得分:6)

使用Subject<T>.HasObservers属性。

Source Code

我不记得它何时被引入,但我很确定它并不总是存在。它可能是在Rx 2.0中添加的。

答案 1 :(得分:5)

您应该尽可能避免实现自己的可观察(或主题)实现。

你当然可以尝试编写一个包装类来帮助。

试试这个:

public class Countable
{
    private int _count;
    public int Count { get { return _count; } }
    public IObservable<T> GetCountable<T>(IObservable<T> source)
    {
        return Observable.Create<T>(o =>
        {
            Interlocked.Increment(ref _count);
            var subscription = source.Subscribe(o);
            var decrement = Disposable.Create(() =>
            {
                Interlocked.Decrement(ref _count);
            });
            return new CompositeDisposable(subscription, decrement);
        });
    }
}

然后您可以编写如下代码:

var xs = new Subject<int>();
var countable = new Countable();
var ys = countable.GetCountable(xs);
Console.WriteLine(countable.Count);
var s1 = ys.Subscribe(y => { });
Console.WriteLine(countable.Count);
var s2 = ys.Subscribe(y => { });
Console.WriteLine(countable.Count);
s1.Dispose();
Console.WriteLine(countable.Count);
s2.Dispose();
Console.WriteLine(countable.Count);

我的结果是:

0
1
2
1
0

答案 2 :(得分:3)

使用subject.observers.length,例如:

import {Subject} from 'rxjs'

let subject = new Subject()
let s1 = subject.subscribe(v => console.log('observerA: ' + v))

subject.next(1) // observerA: 1
console.log(subject.observers.length) // 1


let s2 = subject.subscribe(v => {
    console.log('observerB: ' + v)
    if(v===3) s2.unsubscribe()
})

subject.next(2) // observerA: 2
console.log(subject.observers.length) // 2

subject.next(3) // observerA: 3
console.log(subject.observers.length) // 1