找回密码
立即注册
搜索
热搜: Java Python Linux Go
发回帖 发新帖

2151

积分

0

好友

287

主题
发表于 1 小时前 | 查看: 3| 回复: 0

SpringBoot整合Canal与RabbitMQ架构示意图

你是否也曾为记录数据变更而烦恼?如果直接在业务代码中埋点,不仅耦合度高,而且难以统一管理。今天,我们介绍一种与业务代码解耦的方案:利用Canal监听MySQL的Binlog,再通过RabbitMQ异步处理变更消息,最终在SpringBoot项目中落地实现。

核心需求与实现思路

我需要在SpringBoot项目中,以一种低侵入的方式记录数据变更。记录内容要包含新数据,如果是更新操作,还得能获取到旧数据。经过调研,发现通过Canal监听MySQL的Binlog可以完美满足需求。

但直接处理变更逻辑可能不够灵活,因此我引入了RabbitMQ作为消息中间件。Canal监听到变化后,将事件投递到RabbitMQ,再由消费者异步处理保存记录等逻辑,实现了监听与处理的解耦。

整体实现步骤

  1. 启动MySQL环境,并开启Binlog日志。
  2. 启动Canal服务,为其配置一个具有复制权限的MySQL账号,使其能以Slave身份连接MySQL。
  3. Canal以TCP模式运行,编写Java客户端代码连接Canal,直接接收Binlog变更事件。
  4. Canal切换为RabbitMQ模式,启动RabbitMQ环境,配置Canal与RabbitMQ的连接,让变更事件通过消息队列传递。
  5. SpringBoot整合RabbitMQ,编写消费者处理来自Canal的变更消息。

环境搭建(基于Docker Compose)

我们使用Docker Compose一键拉起MySQL、RabbitMQ和Canal-Server服务。

version: "3"
services:
    mysql:
        network_mode: mynetwork
        container_name: mymysql
        ports:
            - 3306:3306
        restart: always
        volumes:
            - /etc/localtime:/etc/localtime
            - /home/mycontainers/mymysql/data:/data
            - /home/mycontainers/mymysql/mysql:/var/lib/mysql
            - /home/mycontainers/mymysql/conf:/etc/mysql
        environment:
            - MYSQL_ROOT_PASSWORD=root
        command:
            --character-set-server=utf8mb4
            --collation-server=utf8mb4_unicode_ci
            --log-bin=/var/lib/mysql/mysql-bin
            --server-id=1
            --binlog-format=ROW
            --expire_logs_days=7
            --max_binlog_size=500M
        image: mysql:5.7.20
    rabbitmq:
        container_name: myrabbit
        ports:
            - 15672:15672
            - 5672:5672
        restart: always
        volumes:
            - /etc/localtime:/etc/localtime
            - /home/mycontainers/myrabbit/rabbitmq:/var/lib/rabbitmq
        network_mode: mynetwork
        environment:
            - RABBITMQ_DEFAULT_USER=admin
            - RABBITMQ_DEFAULT_PASS=123456
        image: rabbitmq:3.8-management
    canal-server:
        container_name: canal-server
        restart: always
        ports:
            - 11110:11110
            - 11111:11111
            - 11112:11112
        volumes:
            - /home/mycontainers/canal-server/conf/canal.properties:/home/admin/canal-server/conf/canal.properties
            - /home/mycontainers/canal-server/conf/instance.properties:/home/admin/canal-server/conf/example/instance.properties
            - /home/mycontainers/canal-server/logs:/home/admin/canal-server/logs
        network_mode: mynetwork
        depends_on:
            - mysql
            - rabbitmq
        image: canal/canal-server:v1.1.5

我们需要重点关注Canal的配置文件,并通过卷映射(volumes)挂载到容器中。主要涉及两个文件:

  • /home/admin/canal-server/conf/canal.properties:Canal服务端全局配置。
  • /home/admin/canal-server/conf/example/instance.properties:Canal实例配置,路径中的example是实例名,需与canal.properties中的配置对应。

关键配置文件详解

1. canal.properties

这是服务端主配置。我们首先以TCP模式运行,因此设置 canal.serverMode = tcp。注意文件末尾的RabbitMQ连接配置,稍后会用到。

################################################
########     common argument    ############
################################################
# tcp bind ip
canal.ip =
# register ip to zookeeper
canal.register.ip =
canal.port = 11111
canal.metrics.pull.port = 11112

...

# tcp, kafka, rocketMQ, rabbitMQ, pulsarMQ
canal.serverMode = tcp

...

################################################
########     destinations    ############
################################################
canal.destinations = canal-exchange
# conf root dir
canal.conf.dir = ../conf
# auto scan instance dir add/remove and start/stop instance
canal.auto.scan = true
canal.auto.scan.interval = 5

...

#################################################
########         RabbitMQ       ############
#################################################
rabbitmq.host = myrabbit
rabbitmq.virtual.host = /
rabbitmq.exchange = canal-exchange
rabbitmq.username = admin
rabbitmq.password = 123456
rabbitmq.deliveryMode =

提示:从注释可见,Canal支持tcp, kafka, rocketMQ, rabbitMQ, pulsarMQ等多种服务模式。

2. instance.properties

这是数据同步实例的配置,核心是配置其要连接的MySQL源。

################################################
# position info
canal.instance.master.address=mymysql:3306
canal.instance.master.journal.name=
canal.instance.master.position=
canal.instance.master.timestamp=
canal.instance.master.gtid=

# username/password
canal.instance.dbUsername=canal
canal.instance.dbPassword=canal
canal.instance.connectionCharset = UTF-8

# table regex
canal.instance.filter.regex=.*\..*

# mq config
canal.mq.topic=canal-routing-key
canal.mq.partition=0

关键配置项说明:

  • canal.instance.master.address:MySQL地址。因为Canal和MySQL都在Docker的mynetwork网络内,所以这里直接使用容器名mymysql和内部端口3306
  • canal.instance.dbUsername/password:Canal连接MySQL使用的账号密码。

3. 创建Canal专用MySQL用户

为了安全,不建议直接使用root账号。我们需要在MySQL中创建一个专供Canal使用的用户。

# 进入MySQL容器
docker exec -it mymysql bash
# 登录MySQL
mysql -uroot -proot

-- 执行SQL
use mysql;
-- 创建用户 canal,密码 canal
create user 'canal'@'%' identified by 'canal';
-- 授予必要权限(主要是复制相关权限)
grant SELECT, REPLICATION SLAVE, REPLICATION CLIENT on *.* to 'canal'@'%';
flush privileges;

创建完成后,可以使用canal/canal进行连接测试。

阶段一:SpringBoot整合Canal(TCP直连模式)

首先,我们以TCP模式启动Canal,并在SpringBoot中编写客户端直接连接。

1. 添加Maven依赖

<canal.version>1.1.5</canal.version>

<!--canal-->
<dependency>
    <groupId>com.alibaba.otter</groupId>
    <artifactId>canal.client</artifactId>
    <version>${canal.version}</version>
</dependency>
<dependency>
    <groupId>com.alibaba.otter</groupId>
    <artifactId>canal.protocol</artifactId>
    <version>${canal.version}</version>
</dependency>

2. 编写Canal客户端组件
这个组件负责连接Canal服务端,订阅数据库变更,并循环拉取、处理消息。

import com.alibaba.otter.canal.client.CanalConnector;
import com.alibaba.otter.canal.client.CanalConnectors;
import com.alibaba.otter.canal.protocol.CanalEntry;
import com.alibaba.otter.canal.protocol.Message;
import org.springframework.stereotype.Component;

import java.net.InetSocketAddress;
import java.util.List;

@Component
public class CanalClient {

    private final static int BATCH_SIZE = 1000;

    public void run() {
        // 创建链接,连接到localhost:11111,实例名为canal-exchange
        CanalConnector connector = CanalConnectors.newSingleConnector(new InetSocketAddress("localhost", 11111), "canal-exchange", "canal", "canal");
        try {
            //打开连接
            connector.connect();
            //订阅所有表
            connector.subscribe(".*\\..*");
            //回滚到未进行ack的地方
            connector.rollback();
            while (true) {
                // 获取指定数量的数据(无确认)
                Message message = connector.getWithoutAck(BATCH_SIZE);
                long batchId = message.getId();
                int size = message.getEntries().size();
                //如果没有数据
                if (batchId == -1 || size == 0) {
                    try {
                        //线程休眠2秒
                        Thread.sleep(2000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                } else {
                    //如果有数据,处理数据
                    printEntry(message.getEntries());
                }
                // 确认此batchId的消息已处理
                connector.ack(batchId);
            }
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            connector.disconnect();
        }
    }

    /**
     * 打印canal server解析binlog获得的实体类信息
     */
    private static void printEntry(List<CanalEntry.Entry> entrys) {
        for (CanalEntry.Entry entry : entrys) {
            if (entry.getEntryType() == CanalEntry.EntryType.TRANSACTIONBEGIN || entry.getEntryType() == CanalEntry.EntryType.TRANSACTIONEND) {
                //跳过事务开始/结束的实体类型
                continue;
            }
            //RowChange对象,包含了一行数据变化的所有特征
            CanalEntry.RowChange rowChage;
            try {
                rowChage = CanalEntry.RowChange.parseFrom(entry.getStoreValue());
            } catch (Exception e) {
                throw new RuntimeException("ERROR # parser of eromanga-event has an error , data:" + entry.toString(), e);
            }
            //获取操作类型:insert/update/delete类型
            CanalEntry.EventType eventType = rowChage.getEventType();
            //打印Header信息
            System.out.println(String.format("================》; binlog[%s:%s] , name[%s,%s] , eventType : %s",
                    entry.getHeader().getLogfileName(), entry.getHeader().getLogfileOffset(),
                    entry.getHeader().getSchemaName(), entry.getHeader().getTableName(),
                    eventType));
            //判断是否是DDL语句
            if (rowChage.getIsDdl()) {
                System.out.println("================》;isDdl: true,sql:" + rowChage.getSql());
            }
            //获取RowChange对象里的每一行数据,打印出来
            for (CanalEntry.RowData rowData : rowChage.getRowDatasList()) {
                //如果是删除语句
                if (eventType == CanalEntry.EventType.DELETE) {
                    printColumn(rowData.getBeforeColumnsList());
                    //如果是新增语句
                } else if (eventType == CanalEntry.EventType.INSERT) {
                    printColumn(rowData.getAfterColumnsList());
                    //如果是更新的语句
                } else {
                    //变更前的数据
                    System.out.println("------->; before");
                    printColumn(rowData.getBeforeColumnsList());
                    //变更后的数据
                    System.out.println("------->; after");
                    printColumn(rowData.getAfterColumnsList());
                }
            }
        }
    }

    private static void printColumn(List<CanalEntry.Column> columns) {
        for (CanalEntry.Column column : columns) {
            System.out.println(column.getName() + " : " + column.getValue() + "    update=" + column.getUpdated());
        }
    }
}

3. 在启动类中运行客户端

@SpringBootApplication
public class BaseApplication implements CommandLineRunner {
    @Autowired
    private CanalClient canalClient;

    @Override
    public void run(String... args) throws Exception {
        canalClient.run();
    }
}

启动SpringBoot应用,此时在MySQL中执行增删改操作,控制台就会打印出详细的变更信息。但这只是第一步,接下来我们将引入RabbitMQ,让架构更优雅。

阶段二:Canal整合RabbitMQ

现在我们改造Canal,让它将变更事件发送到RabbitMQ,而非直接由TCP客户端处理。

1. 修改Canal配置

  • canal.properties 中,将服务模式改为 rabbitMQ
    canal.serverMode = rabbitMQ
  • 确保RabbitMQ连接配置正确(我们在docker-compose中已配置)。
  • instance.properties 中,确认 canal.mq.topic 的值(它将成为RabbitMQ的routing key)。
    canal.mq.topic=canal-routing-key

2. 重启Canal-Server容器
配置修改完成后,重启Canal-Server容器使配置生效。

3. (可选)在RabbitMQ管理页面创建交换机和队列
虽然SpringBoot可以自动声明,但手动创建有助于理解流程。

  • 创建名为 canal-exchange 的交换机(Direct类型)。
  • 创建名为 canal-queue 的队列。
  • 将队列绑定到交换机,并设置 routing-keycanal-routing-key

阶段三:SpringBoot整合RabbitMQ消费消息

现在,我们将编写SpringBoot应用作为RabbitMQ的消费者,处理来自Canal的变更消息。

1. 添加依赖

<amqp.version>2.3.4.RELEASE</amqp.version>

<!--消息队列-->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-amqp</artifactId>
    <version>${amqp.version}</version>
</dependency>

2. 配置RabbitMQ连接(application.yml)

spring:
  rabbitmq:
    host: 192.168.0.108 # 你的RabbitMQ服务器IP
    port: 5672
    username: admin
    password: 123456
    # 消息确认配置
    publisher-confirm-type: correlated
    publisher-returns: true

3. RabbitMQ配置类

@Configuration
public class RabbitConfig {
    @Bean
    public RabbitTemplate rabbitTemplate(ConnectionFactory connectionFactory) {
        RabbitTemplate template = new RabbitTemplate();
        template.setConnectionFactory(connectionFactory);
        template.setMessageConverter(new Jackson2JsonMessageConverter());
        return template;
    }

    /**
     * 配置Listener容器工厂,解决消息转换问题
     */
    @Bean
    public SimpleRabbitListenerContainerFactory rabbitListenerContainerFactory(ConnectionFactory connectionFactory) {
        SimpleRabbitListenerContainerFactory factory = new SimpleRabbitListenerContainerFactory();
        factory.setConnectionFactory(connectionFactory);
        factory.setMessageConverter(new Jackson2JsonMessageConverter());
        return factory;
    }
}

4. 声明队列、交换机及绑定(生产者配置)

public class RabbitConstant {
    public static final String CanalQueue = "canal-queue";
    public static final String CanalExchange = "canal-exchange";
    public static final String CanalRouting = "canal-routing-key";
}

@Configuration
public class CanalProvider {
    /**
     * 声明队列
     */
    @Bean
    public Queue canalQueue() {
        return new Queue(RabbitConstant.CanalQueue, true);
    }
    /**
     * 声明直连交换机
     */
    @Bean
    DirectExchange canalExchange() {
        return new DirectExchange(RabbitConstant.CanalExchange, true, false);
    }
    /**
     * 绑定队列到交换机
     */
    @Bean
    Binding bindingCanal() {
        return BindingBuilder.bind(canalQueue()).to(canalExchange()).with(RabbitConstant.CanalRouting);
    }
}

5. 消息消费者
这是最终处理数据变更逻辑的地方。

/**
 * Canal消息消费者
 */
@Component
@RabbitListener(queues = RabbitConstant.CanalQueue)
public class CanalComsumer {
    private final SysBackupService sysBackupService;

    public CanalComsumer(SysBackupService sysBackupService) {
        this.sysBackupService = sysBackupService;
    }

    @RabbitHandler
    public void process(Map<String, Object> msg) {
        System.out.println("收到canal消息:" + msg);
        boolean isDdl = (boolean) msg.get("isDdl");
        // 不处理DDL事件
        if (isDdl) {
            return;
        }
        // 消息ID
        int tid = (int) msg.get("id");
        // 时间戳
        long ts = (long) msg.get("ts");
        // 数据库名
        String database = (String) msg.get("database");
        // 表名
        String table = (String) msg.get("table");
        // 操作类型:INSERT/UPDATE/DELETE
        String type = (String) msg.get("type");
        // 新数据
        List<?> data = (List<?>) msg.get("data");
        // 旧数据(仅UPDATE操作有值)
        List<?> old = (List<?>) msg.get("old");

        // 示例:跳过备份表自身产生的变更,防止无限循环
        if ("sys_backup".equalsIgnoreCase(table)) {
            return;
        }
        // 只处理需要的操作类型
        if (!"INSERT".equalsIgnoreCase(type)
                && !"UPDATE".equalsIgnoreCase(type)
                && !"DELETE".equalsIgnoreCase(type)) {
            return;
        }
        // TODO: 在这里编写你的业务逻辑,例如将 data 和 old 保存到另一张记录表
        // sysBackupService.saveBackupRecord(...);
    }
}

测试与总结

完成以上所有步骤后,启动整个系统设计

  1. 运行 docker-compose up 启动基础设施。
  2. 启动你的SpringBoot应用。
  3. 在MySQL中任意修改一条数据。
  4. 观察SpringBoot应用的控制台,将会打印出从RabbitMQ接收到的、格式化的JSON变更消息。

至此,一个基于SpringBoot、Canal和RabbitMQ的数据库变更监听与异步处理方案就完整实现了。这套方案将数据变更的捕获与业务处理完全解耦,扩展性强,非常适合用于数据同步、审计日志、缓存更新等场景。如果你在实践过程中遇到问题,欢迎到云栈社区与大家交流探讨。




上一篇:iPhone 17换屏退货事件调查:苹果官网审核收紧,二手配件灰色产业链浮现
下一篇:从大厂P6面试失败,看平台光环下的个人能力认知误区
您需要登录后才可以回帖 登录 | 立即注册

手机版|小黑屋|网站地图|云栈社区 ( 苏ICP备2022046150号-2 )

GMT+8, 2026-3-10 22:32 , Processed in 0.537121 second(s), 41 queries , Gzip On.

Powered by Discuz! X3.5

© 2025-2026 云栈社区.

快速回复 返回顶部 返回列表