我们看到发布到ActionBlock时出现意外行为,即使MaxDegreeOfParallelism为1,也会发生并行处理。这是场景。
发布到ActionBlock的类看起来像这样:
public class ByteHandler {
...
public ByteHandler(ByteHandlingCallback delegateCallback){
_byteHandlingActionBlock = new ActionBlock<byte[]>(bytes => delegateCallback.HandleBytes(bytes));
}
public void HandleBytes(byte[] bytes){
_byteHandlingActionBlock.Post(bytes);
}
下游,我们将字节反序列化为对象,并将这些对象(让我们称之为Notifications)传递给处理程序,具体取决于它们的类型:
public class NotificationHandler{
private readonly Dictionary<string, AutoResetEvent> _secondNoticeReceivedEvents;
public void HandleInitialNotification(InitialNotification notification){
var secondNoticeReceivedEvent = new AutoResetEvent(false);
if (!_secondNoticeReceivedEvents.TryAdd(notification.ID, secondNoticeReceivedEvent)) return;
DoSomethingDownstream(notification);
if (secondNoticeReceivedEvent.WaitOne(_timeout))
DoSomethingElseDownstream();
else
throw new Exception("Second notification not received within timeout!");
_secondNoticeReceivedEvents.TryRemove(notification.ID, out secondNoticeReceivedEvent);
}
public void HandleSecondNotification(SecondNotification notification){
AutoResetEvent secondNoticeReceivedEvent;
if (_secondNoticeReceivedEvents.TryRemove(notification.ID, out secondNoticeReceivedEvent))
secondNoticeReceivedEvent.Set();
}
此处理程序有一个致命的错误:InitialNotifications在它们相应的SecondNotifications之前进入,但HandleInitialNotification在退出之前等待HandleSecondNotification,因此该线程永远不会到达HandleSecondNotification。
通常,我们看到HandleInitialNotification阻塞,直到它超时等待HandleSecondNotification,然后继续执行,同一线程上处理待处理的SecondNotification。这是我们通常在日志中看到的内容:
2013-07-05 13:27:25,755 [13] INFO Received InitialNotification for: XX
2013-07-05 13:27:35,758 [13] WARN Second notification not not received for XX within timeout!
2013-07-05 13:27:35,761 [13] INFO Received SecondNotification for: XX
这不是代码的工作方式,但考虑到它的编写方式,它应该总是超时等待SecondNotification。但是,我们偶尔也会看到HandleInitialNotification在超时之前完成,HandleSecondNotification会在另一个线程上及时处理:
2013-07-05 13:38:13,232 [56] INFO Received InitialNotification for: YY
2013-07-05 13:38:13,258 [11] INFO Received SecondNotification for: YY
由于我们使用的是默认的ActionBlock,因此MaxDegreeOfParallelism应该是1.那么,当发布到ActionBlock的原始线程阻塞时,第二个线程(源自ActionBlock)是否可以获取SecondNotification? / p>
答案 0 :(得分:0)
请注意,您的超时/警告几乎在10秒内发生,而所有其他事件都在几毫秒内发生。您很可能以某种方式同步调用通知处理程序方法(HandleInitialNotification&amp; HandleSecondNotification),以便actionblock在actionblock抓取其缓冲区中的下一个项目之前等待HandleInitialNotification中的超时完成。您需要异步执行超时等待。