The Google Code project does reference a technical paper on the implementation of the ring buffer, however it is a bit dry, academic and tough going for someone wanting to learn how it works. However there are some blog posts that have started to explain the internals in a more readable way. There is an explanation of ring buffer that is the core of the disruptor pattern, a description of the consumer barriers (the part related to reading from the disruptor) and some information on handling multiple producers available.
对中断器最简单的描述是:它是一种在线程之间以最有效的方式发送消息的方法。它可以用作队列的替代品,但它还与SEDA和actor共享许多特性。
与队列相比:
Disruptor提供了将消息传递到另一个线程的能力,并在需要时将其唤醒(类似于BlockingQueue)。然而,有3个明显的区别。
The user of the Disruptor defines how messages are stored by extending Entry class and providing a factory to do the preallocation. This allows for either memory reuse (copying) or the Entry could contain a reference to another object.
Putting messages into the Disruptor is a 2-phase process, first a slot is claimed in the ring buffer, which provides the user with the Entry that can be filled with the appropriate data. Then the entry must be committed, this 2-phase approach is necessary to allow for the flexible use of memory mentioned above. It is the commit that makes the message visible to the consumer threads.
It is the responsibility of the consumer to keep track of the messages that have been consumed from the ring buffer. Moving this responsibility away from the ring buffer itself helped reduce the amount of write contention as each thread maintains its own counter.
与演员相比
Actor模型比大多数其他编程模型更接近于Disruptor,特别是如果您使用所提供的BatchConsumer/BatchHandler类时。这些类隐藏了维护所使用的序列号的所有复杂性,并在重要事件发生时提供了一组简单的回调。然而,有一些细微的区别。
The Disruptor uses a 1 thread - 1 consumer model, where Actors use an N:M model i.e. you can have as many actors as you like and they will be distributed across a fixed numbers of threads (generally 1 per core).
The BatchHandler interface provides an additional (and very important) callback onEndOfBatch(). This allows for slow consumers, e.g. those doing I/O to batch events together to improve throughput. It is possible to do batching in other Actor frameworks, however as nearly all other frameworks don't provide a callback at the end of the batch you need to use a timeout to determine the end of the batch, resulting in poor latency.
与SEDA相比
LMAX构建了破坏者模式来取代基于SEDA的方法。
The main improvement that it provided over SEDA was the ability to do work in parallel. To do this the Disruptor supports multi-casting the same messages (in the same order) to multiple consumers. This avoids the need for fork stages in the pipeline.
We also allow consumers to wait on the results of other consumers without having to put another queuing stage between them. A consumer can simply watch the sequence number of a consumer that it is dependent on. This avoids the need for join stages in pipeline.
与记忆障碍相比
另一种思考方式是,它是一个结构化的,有序的记忆屏障。其中生产者屏障形成写屏障,消费者屏障是读屏障。