为什么我不需要在这个冷的可观察面上发布?

时间:2011-11-12 19:59:19

标签: c# system.reactive reactive-programming observable

由于我感冒了Observable而且我多次订阅“分组”,为什么我不需要在这里发布?当我运行它时,我会期望它会带来不必要的结果,但令我惊讶的是它可以使用和不使用Publish。那是为什么?

var subject = new List<string>
    {                            
    "test",                        
    "test",                 
    "hallo",
    "test",
    "hallo"
    }.ToObservable();
subject
    .GroupBy(x => x)
    .SelectMany(grouped => grouped.Scan(0, (count, _) => ++count)
         .Zip(grouped, (count, chars) => new { Chars = chars, Count = count }))
    .Subscribe(result => Console.WriteLine("You typed {0} {1} times", 
         result.Chars, result.Count));

// I Would have expect that I need to use Publish like that
//subject
//   .GroupBy(x => x)
//   .SelectMany(grouped => grouped.Publish(sharedGroup => 
//       sharedGroup.Scan(0, (count, _) => ++count)
//       .Zip(sharedGroup, (count, chars) => 
//           new { Chars = chars, Count = count })))
//   .Subscribe(result => Console.WriteLine("You typed {0} {1} times", 
//       result.Chars, result.Count));

Console.ReadLine();

修改

保罗注意到,因为我们订阅了两次潜在的冷观察,我们应该重复两次。但是,我没有运气使这个效果可见。我试图插入调试行,但例如,这只打印“执行”一次。

var subject = new List<Func<string>>
{                            
() =>
    {
        Console.WriteLine("performing");
        return "test";
    },                        
() => "test",                 
() => "hallo",
() => "test",
() => "hallo"
}.ToObservable();


subject
    .Select(x => x())
    .GroupBy(x => x)
    .SelectMany(grouped => grouped.Scan(0, (count, _) => ++count)
            .Zip(grouped, (count, chars) => new { Chars = chars, Count = count }))
    .Subscribe(result => Console.WriteLine("You typed {0} {1} times",
            result.Chars, result.Count));

我想知道我们是否可以使效果可见,因为我们正在处理冷可观察并且没有使用Publish()。在另一个步骤中,我想看看Publish()(见上文)如何使效果消失。

编辑2

正如Paul所说,我为调试目的创建了一个自定义IObservable<string>。但是,如果您使用Subscribe()方法设置断点,您会注意到它只会被点击

class Program
{
    static void Main(string[] args)
    {
        var subject = new MyObservable();

        subject
            .GroupBy(x => x)
            .SelectMany(grouped => grouped.Scan(0, (count, _) => ++count)
                 .Zip(grouped, (count, chars) => new { Chars = chars, Count = count }))
            .Subscribe(result => Console.WriteLine("You typed {0} {1} times",
                 result.Chars, result.Count));

       Console.ReadLine();
   }
}

class MyObservable : IObservable<string>
{
    public IDisposable Subscribe(IObserver<string> observer)
    {
        observer.OnNext("test");
        observer.OnNext("test");
        observer.OnNext("hallo");
        observer.OnNext("test");
        observer.OnNext("hallo");
        return Disposable.Empty;
    }
}

所以对我来说问题仍然存在。为什么我在这个寒冷的Publish上不需要Observable

2 个答案:

答案 0 :(得分:6)

您只使用基于列表的源一次,因此您不会在那里看到重复的订阅效果。回答你问题的关键是以下观察:

  

IGroupedObservable&lt; K,T&gt;流出GroupBy的对象本身就是伪装的主题。

在内部,GroupBy保持字典&lt; K,ISubject&lt; T&gt;&gt;。每当有消息进入时,它就会被相应的密钥发送到主题中。您正在订阅两次分组对象,这是安全的,因为主题将生产者与消费者分离。

答案 1 :(得分:1)

在Zip中重复使用'groups'意味着您有效地对每个分组进行了两次 - 但是,由于您的源是Cold,它仍然有效。这有意义吗?