Canal数据同步多库循环问题,修改源码解决

文章讲述了作者在使用Canal进行数据库同步时遇到的问题,包括1.1.7版本与MySQL8.0.3x的兼容性问题,以及如何通过修改源码处理主子库循环同步和DML/DDL操作。作者提到通过query方法对比数据差异并进行更新和插入操作,同时处理DDL的异常和数据结构同步。
摘要由CSDN通过智能技术生成

用canal做数据同步,业务上分为主子库,相互同步,这样就会出现循环同步的情况。

因为用的开源代码,最简单的办法就是修改源码。

下载最新1.1.7版本代码。

这里遇到了几个问题:

  • 我的MySQL版本是8.0.3x,之前用1.1.6版本链接会提示链接错误,更换1.1.7版本解决。
  • 编辑器是idea,在启动1.1.7版本代码,总是提示错误:
    Could not initialize class com.sun.org.apache.xml.internal.serializer.Encodings

网上查到问题点如下:链接

解决方案,从jdk8换成了jdk17,解决了。

canal有两个入口,一个是deployer模块下的CanalLauncher,一个是client-adaptor的launcher模块下的CanalAdapterApplication,至于admin的就很简单了。

剩下的就是修改源码了。

我这边多库同步是通过的rdb进行的,这里就涉及到两个部分:

DML:数据内容的同步,这里主要针对insert和update,因为不涉及大量数据吞吐的情况,所以通过最简单的query查询,然后去比对是否相同,来判断是否再进行处理。



    /**
     * 插入操作
     *
     * @param config 配置项
     * @param dml DML数据
     */
    private ResultSet query(BatchExecutor batchExecutor, MappingConfig config, SingleDml dml) throws SQLException {

        Map<String, Object> data = dml.getData();
        DbMapping dbMapping = config.getDbMapping();
        Map<String, Integer> ctype = getTargetColumnType(batchExecutor.getConn(), config);
        StringBuilder sql = new StringBuilder();
        sql.append("select * FROM ").append(SyncUtil.getDbTableName(dbMapping, dataSource.getDbType())).append(" WHERE ");

        List<Map<String, ?>> values = new ArrayList<>();
        // 拼接主键
        appendCondition(dbMapping, sql, ctype, values, data);
        logger.info("execute query sql => {}",sql);
        return batchExecutor.query(sql.toString(),values);
    }


    insert(){

        //进行校验
        ResultSet resultSet = query(batchExecutor,config,dml);
        boolean hasNext = resultSet.next();
        resultSet.close();
        if(hasNext){
            logger.warn("insert data already exists,pls check!");
            return;
        }

        ......

}


update(){

ResultSet currentRes = query(batchExecutor,config,dml);
        boolean updateFlag = false;
        if (currentRes.next()){
            for (String srcColumnName : old.keySet()) {
                if(!currentRes.getString(srcColumnName).equalsIgnoreCase(data.get(srcColumnName).toString())){
                    updateFlag = true;
                    break;
                }
            }
        }else{
            logger.warn("The UPDATE data is not correct,pls check");
        }
        currentRes.close();
        if(!updateFlag){
            logger.warn("The UPDATE data not changed,no need to update");
            return;
        }


}

DDL:数据结构的同步

这里做了异常捕获处理,分析错误原因,判断是否继续进行操作。

然后给rdb打包,替代原来的就可以了。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值