分区详解-canal.mq.partitionHash

1 篇文章 0 订阅
1 篇文章 0 订阅

canal.mq.partitionHash

实际配置使用
// 共16个queue
canal.mq.partitionsNum=16
// 根据policyNo分到不同的queue,以此来保证顺序消费
canal.mq.partitionHash=ins_core_fccb_db.prpcmain:policyno,ins_core_fccb_db.prpcinsured:policyno
分配算法
com.alibaba.otter.canal.connector.rocketmq.producer.CanalRocketMQProducer#send
-->
com.alibaba.otter.canal.connector.core.producer.MQMessageUtils#messagePartition
1. 如果是ddl,则直接默认放到第一个queue里面
if (flatMessage.getIsDdl()) {
	partitionMessages[0] = flatMessage;
}
2. 根据库名、表名和canal.mq.partitionHash获取HashMode
com.alibaba.otter.canal.connector.core.producer.MQMessageUtils#getPartitionHashColumns

2.1. 如果都没有匹配HashMode,发送到第一个分区
partitionMessages[0] = flatMessage;
2.2. 如果pkHashConfig.lastIndexOf(“:”)<0则认为是根据表名进行分配
if (hashMode.tableHash) {
	int hashCode = table.hashCode();
	int pkHash = Math.abs(hashCode) % partitionsNum;
	// math.abs可能返回负值,这里再取反,把出现负值的数据还是写到固定的分区,仍然可以保证消费顺序
	pkHash = Math.abs(pkHash);
	partitionMessages[pkHash] = flatMessage;
}
2.3. 根据设置的列进行分配
for (Map<String, String> row : flatMessage.getData()) {
	int hashCode = 0;
	if (databaseHash) {
		hashCode = database.hashCode();
	}
	if (pkNames != null) {
		for (String pkName : pkNames) {
			String value = row.get(pkName);
			if (value == null) {
				value = "";
			}
			hashCode = hashCode ^ value.hashCode();
		}
	}

	int pkHash = Math.abs(hashCode) % partitionsNum;
	// math.abs可能返回负值,这里再取反,把出现负值的数据还是写到固定的分区,仍然可以保证消费顺序
	pkHash = Math.abs(pkHash);

	FlatMessage flatMessageTmp = partitionMessages[pkHash];
	...
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值