CopyOnWrite 思想在 Kafka 源码中的运用服务器
在 Kafka 的内核源码中,有这么一个场景,客户端在向 Kafka 写数据的时候,会把消息先写入客户端本地的内存缓冲,而后在内存缓冲里造成一个 Batch 以后再一次性发送到 Kafka 服务器上去,这样有助于提高吞吐量。
请看下图:数据结构
这个时候 Kafka 的内存缓冲用的是什么数据结构呢?
请看源码:并发
private final ConcurrentMap<TopicPartition, Deque<RecordBatch>> batches = new CopyOnWriteMap<TopicPartition, Deque<RecordBatch>>();
这个数据结构就是核心的用来存放写入内存缓冲中的消息的数据结构,要看懂这个数据结构须要对不少 Kafka 内核源码里的概念进行解释。Kafka 是本身实现了一个 CopyOnWriteMap,这个CopyOnWriteMap 采用的就是 CopyOnWrite 思想。
咱们来看一下这个 CopyOnWriteMap 的源码实现:ide
// 典型的volatile修饰普通Map private volatile Map<K, V> map; @Override public synchronized V put(K k, V v) { // 更新的时候先建立副本,更新副本,而后对volatile变量赋值写回去 Map<K, V> copy = new HashMap<K, V>(this.map); V prev = copy.put(k, v); this.map = Collections.unmodifiableMap(copy); return prev; } @Override public V get(Object k) { // 读取的时候直接读volatile变量引用的map数据结构,无需锁 return map.get(k); }
Kafka 这个核心数据结构在这里之因此采用 CopyOnWriteMap 思想来实现,就是由于这个 Map 的 Key-Value 对,其实没那么频繁更新。性能
也就是 TopicPartition-Deque 这个 Key-Value 对,更新频率很低。可是它的 Get 操做倒是高频的读取请求,由于会高频的读取出来一个 TopicPartition 对应的 Deque 数据结构,来对这个队列进行入队出队等操做,因此对于这个 Map 而言,高频的是其 Get 操做。这个时候,Kafka 就采用了 CopyOnWrite 思想来实现这个 Map,避免更新 Key-Value 的时候阻塞住高频的读操做,实现无锁的效果,优化线程并发的性能。优化
相信看完这个文章,对于 CopyOnWrite 思想以及适用场景,包括 JDK 中的实现,以及在 Kafka 源码中的运用,都有了一个切身的体会了。this