精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

SpringBoot整合Canal、RabbitMQ監(jiān)聽數(shù)據(jù)變更~

開發(fā) 前端
經(jīng)過調(diào)研發(fā)現(xiàn),使用Canal來監(jiān)聽MySQL的binlog變化可以實現(xiàn)這個需求,可是在監(jiān)聽到變化后需要馬上保存變更記錄,除非再做一些邏輯處理,于是我又結(jié)合了RabbitMQ來處理保存變更記錄的操作。

需求

我想要在SpringBoot中采用一種與業(yè)務(wù)代碼解耦合的方式,來實現(xiàn)數(shù)據(jù)的變更記錄,記錄的內(nèi)容是新數(shù)據(jù),如果是更新操作還得有舊數(shù)據(jù)內(nèi)容。

經(jīng)過調(diào)研發(fā)現(xiàn),使用Canal來監(jiān)聽MySQL的binlog變化可以實現(xiàn)這個需求,可是在監(jiān)聽到變化后需要馬上保存變更記錄,除非再做一些邏輯處理,于是我又結(jié)合了RabbitMQ來處理保存變更記錄的操作。

步驟

  • 啟動MySQL環(huán)境,并開啟binlog
  • 啟動Canal環(huán)境,為其創(chuàng)建一個MySQL賬號,然后以Slave的形式連接MySQL
  • Canal服務(wù)模式設(shè)為TCP,用Java編寫客戶端代碼,監(jiān)聽MySQL的binlog修改
  • Canal服務(wù)模式設(shè)為RabbitMQ,啟動RabbitMQ環(huán)境,配置Canal和RabbitMQ的連接,用消息隊列去接收binlog修改事件

環(huán)境搭建

環(huán)境搭建基于docker-compose:

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
            # - canal-admin
        image: canal/canal-server:v1.1.5

我們需要修改下Canal環(huán)境的配置文件:canal.properties和instance.properties,映射Canal中的以下兩個路徑:

  • /home/admin/canal-server/conf/canal.properties配置文件中,canal.destinations意思是server上部署的instance列表,
  • /home/admin/canal-server/conf/example/instance.properties這里的/example是指instance即實例名,要和上面canal.properties內(nèi)instance配置對應(yīng),canal會為實例創(chuàng)建對應(yīng)的文件夾,一個Client對應(yīng)一個實例

以下是我們需要準備的兩個配置文件具體內(nèi)容:

canal.properties

#################################################
#########     common argument   #############
#################################################
# tcp bind ip
canal.ip =
# register ip to zookeeper
canal.register.ip =
canal.port = 11111
canal.metrics.pull.port = 11112
# canal instance user/passwd
# canal.user = canal
# canal.passwd = E3619321C1A937C46A0D8BD1DAC39F93B27D4458

# canal admin config
# canal.admin.manager = canal-admin:8089

# canal.admin.port = 11110
# canal.admin.user = admin
# canal.admin.passwd = 6BB4837EB74329105EE4568DDA7DC67ED2CA2AD9

# admin auto register 自動注冊
# canal.admin.register.auto = true
# 集群名,單機則不寫
# canal.admin.register.cluster =
# Canal Server 名字
# canal.admin.register.name = canal-admin

canal.zkServers =
# flush data to zk
canal.zookeeper.flush.period = 1000
canal.withoutNetty = false
# tcp, kafka, rocketMQ, rabbitMQ, pulsarMQ
canal.serverMode = tcp
# flush meta cursor/parse position to file
canal.file.data.dir = ${canal.conf.dir}
canal.file.flush.period = 1000
## memory store RingBuffer size, should be Math.pow(2,n)
canal.instance.memory.buffer.size = 16384
## memory store RingBuffer used memory unit size , default 1kb
canal.instance.memory.buffer.memunit = 1024 
## meory store gets mode used MEMSIZE or ITEMSIZE
canal.instance.memory.batch.mode = MEMSIZE
canal.instance.memory.rawEntry = true

## detecing config
canal.instance.detecting.enable = false
#canal.instance.detecting.sql = insert into retl.xdual values(1,now()) on duplicate key update x=now()
canal.instance.detecting.sql = select 1
canal.instance.detecting.interval.time = 3
canal.instance.detecting.retry.threshold = 3
canal.instance.detecting.heartbeatHaEnable = false

# support maximum transaction size, more than the size of the transaction will be cut into multiple transactions delivery
canal.instance.transaction.size =  1024
# mysql fallback connected to new master should fallback times
canal.instance.fallbackIntervalInSeconds = 60

# network config
canal.instance.network.receiveBufferSize = 16384
canal.instance.network.sendBufferSize = 16384
canal.instance.network.soTimeout = 30

# binlog filter config
canal.instance.filter.druid.ddl = true
canal.instance.filter.query.dcl = false
canal.instance.filter.query.dml = false
canal.instance.filter.query.ddl = false
canal.instance.filter.table.error = false
canal.instance.filter.rows = false
canal.instance.filter.transaction.entry = false
canal.instance.filter.dml.insert = false
canal.instance.filter.dml.update = false
canal.instance.filter.dml.delete = false

# binlog format/image check
canal.instance.binlog.format = ROW,STATEMENT,MIXED 
canal.instance.binlog.image = FULL,MINIMAL,NOBLOB

# binlog ddl isolation
canal.instance.get.ddl.isolation = false

# parallel parser config
canal.instance.parser.parallel = true
## concurrent thread number, default 60% available processors, suggest not to exceed Runtime.getRuntime().availableProcessors()
canal.instance.parser.parallelThreadSize = 16
## disruptor ringbuffer size, must be power of 2
canal.instance.parser.parallelBufferSize = 256

# table meta tsdb info
canal.instance.tsdb.enable = true
canal.instance.tsdb.dir = ${canal.file.data.dir:../conf}/${canal.instance.destination:}
canal.instance.tsdb.url = jdbc:h2:${canal.instance.tsdb.dir}/h2;CACHE_SIZE=1000;MODE=MYSQL;
canal.instance.tsdb.dbUsername = canal
canal.instance.tsdb.dbPassword = canal
# dump snapshot interval, default 24 hour
canal.instance.tsdb.snapshot.interval = 24
# purge snapshot expire , default 360 hour(15 days)
canal.instance.tsdb.snapshot.expire = 360

#################################################
#########     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
# set this value to 'true' means that when binlog pos not found, skip to latest.
# WARN: pls keep 'false' in production env, or if you know what you want.
canal.auto.reset.latest.pos.mode = false

canal.instance.tsdb.spring.xml = classpath:spring/tsdb/h2-tsdb.xml
#canal.instance.tsdb.spring.xml = classpath:spring/tsdb/mysql-tsdb.xml

canal.instance.global.mode = spring
canal.instance.global.lazy = false
canal.instance.global.manager.address = ${canal.admin.manager}
#canal.instance.global.spring.xml = classpath:spring/memory-instance.xml
canal.instance.global.spring.xml = classpath:spring/file-instance.xml
#canal.instance.global.spring.xml = classpath:spring/default-instance.xml

##################################################
#########         MQ Properties      #############
##################################################
# aliyun ak/sk , support rds/mq
canal.aliyun.accessKey =
canal.aliyun.secretKey =
canal.aliyun.uid=

canal.mq.flatMessage = true
canal.mq.canalBatchSize = 50
canal.mq.canalGetTimeout = 100
# Set this value to "cloud", if you want open message trace feature in aliyun.
canal.mq.accessChannel = local

canal.mq.database.hash = true
canal.mq.send.thread.size = 30
canal.mq.build.thread.size = 8

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

此時canal.serverMode = tcp,即TCP直連,我們先開啟這個服務(wù),然后手寫Java客戶端代碼去連接它,等下再改為RabbitMQ。

通過注釋可以看到,canal支持的服務(wù)模式有:tcp, kafka, rocketMQ, rabbitMQ, pulsarMQ,即主流的消息隊列都支持。

instance.properties

#################################################
## mysql serverId , v1.0.26+ will autoGen
#canal.instance.mysql.slaveId=123

# enable gtid use true/false
canal.instance.gtidon=false

# 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=

# rds oss binlog
canal.instance.rds.accesskey=
canal.instance.rds.secretkey=
canal.instance.rds.instanceId=

# table meta tsdb info
canal.instance.tsdb.enable=true
#canal.instance.tsdb.url=jdbc:mysql://127.0.0.1:3306/canal_tsdb
#canal.instance.tsdb.dbUsername=canal
#canal.instance.tsdb.dbPassword=canal

#canal.instance.standby.address =
#canal.instance.standby.journal.name =
#canal.instance.standby.position =
#canal.instance.standby.timestamp =
#canal.instance.standby.gtid=

# username/password
canal.instance.dbUsername=canal
canal.instance.dbPassword=canal
canal.instance.connectionCharset = UTF-8
# enable druid Decrypt database password
canal.instance.enableDruid=false
#canal.instance.pwdPublicKey=MFwwDQYJKoZIhvcNAQEBBQADSwAwSAJBALK4BUxdDltRRE5/zXpVEVPUgunvscYFtEip3pmLlhrWpacX7y7GCMo2/JM6LeHmiiNdH1FWgGCpUfircSwlWKUCAwEAAQ==

# table regex
canal.instance.filter.regex=.*\..*
# table black regex
canal.instance.filter.black.regex=mysql\.slave_.*
# table field filter(format: schema1.tableName1:field1/field2,schema2.tableName2:field1/field2)
#canal.instance.filter.field=test1.t_product:id/subject/keywords,test2.t_company:id/name/contact/ch
# table field black filter(format: schema1.tableName1:field1/field2,schema2.tableName2:field1/field2)
#canal.instance.filter.black.field=test1.t_product:subject/product_image,test2.t_company:id/name/contact/ch

# mq config
canal.mq.topic=canal-routing-key
# dynamic topic route by schema or table regex
#canal.mq.dynamicTopic=mytest1.user,topic2:mytest2\..*,.*\..*
canal.mq.partition=0

把這兩個配置文件映射好,再次提醒,注意實例的路徑名,默認是:/example/instance.properties

修改canal配置文件

我們需要修改這個實例配置文件,去連接MySQL,確保以下的配置正確:

canal.instance.master.address=mymysql:3306
canal.instance.dbUsername=canal
canal.instance.dbPassword=canal

mymysql是同為docker容器的MySQL環(huán)境,端口3306是指內(nèi)部端口。

這里多說明一下,docker端口配置時假設(shè)為:13306:3306,那么容器對外的端口就是13306,內(nèi)部是3306,在本示例中,MySQL和Canal都是容器環(huán)境,所以Canal連接MySQL需要滿足以下條件:

  • 處于同一網(wǎng)段(docker-compose.yml中的mynetwork)
  • 訪問內(nèi)部端口(即3306,而非13306)

dbUsername和dbPassword為MySQL賬號密碼,為了開發(fā)方便可以使用root/root,但是我仍建議自行創(chuàng)建用戶并分配訪問權(quán)限:

# 進入docker中的mysql容器
docker exec -it mymysql bash
# 進入mysql指令模式
mysql -uroot -proot

# 編寫MySQL語句并執(zhí)行
> ...
-- 選擇mysql
use mysql;
-- 創(chuàng)建canal用戶,賬密:canal/canal
create user 'canal'@'%' identified by 'canal';
-- 分配權(quán)限,以及允許所有主機登錄該用戶
grant SELECT, INSERT, UPDATE, DELETE, REPLICATION SLAVE, REPLICATION CLIENT on *.* to 'canal'@'%';

-- 刷新一下使其生效
flush privileges;

-- 附帶一個刪除用戶指令
drop user 'canal'@'%';

用navicat或者shell去登錄canal這個用戶,可以訪問即創(chuàng)建成功

整合SpringBoot Canal實現(xiàn)客戶端

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>
復(fù)制代碼

新增組件并啟動:

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.boot.CommandLineRunner;
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() {
        // 創(chuàng)建鏈接
        CanalConnector connector = CanalConnectors.newSingleConnector(new InetSocketAddress("localhost", 11111), "canal-exchange", "canal", "canal");
        try {
            //打開連接
            connector.connect();
            //訂閱數(shù)據(jù)庫表,全部表
            connector.subscribe(".*\..*");
            //回滾到未進行ack的地方,下次fetch的時候,可以從最后一個沒有ack的地方開始拿
            connector.rollback();
            while (true) {
                // 獲取指定數(shù)量的數(shù)據(jù)
                Message message = connector.getWithoutAck(BATCH_SIZE);
                //獲取批量ID
                long batchId = message.getId();
                //獲取批量的數(shù)量
                int size = message.getEntries().size();
                //如果沒有數(shù)據(jù)
                if (batchId == -1 || size == 0) {
                    try {
                        //線程休眠2秒
                        Thread.sleep(2000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                } else {
                    //如果有數(shù)據(jù),處理數(shù)據(jù)
                    printEntry(message.getEntries());
                }
                //進行 batch id 的確認。確認之后,小于等于此 batchId 的 Message 都會被確認。
                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) {
                //開啟/關(guān)閉事務(wù)的實體類型,跳過
                continue;
            }
            //RowChange對象,包含了一行數(shù)據(jù)變化的所有特征
            //比如isDdl 是否是ddl變更操作 sql 具體的ddl sql beforeColumns afterColumns 變更前后的數(shù)據(jù)字段等等
            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對象里的每一行數(shù)據(jù),打印出來
            for (CanalEntry.RowData rowData : rowChage.getRowDatasList()) {
                //如果是刪除語句
                if (eventType == CanalEntry.EventType.DELETE) {
                    printColumn(rowData.getBeforeColumnsList());
                    //如果是新增語句
                } else if (eventType == CanalEntry.EventType.INSERT) {
                    printColumn(rowData.getAfterColumnsList());
                    //如果是更新的語句
                } else {
                    //變更前的數(shù)據(jù)
                    System.out.println("------->; before");
                    printColumn(rowData.getBeforeColumnsList());
                    //變更后的數(shù)據(jù)
                    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());
        }
    }
}

啟動類Application:

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

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

啟動程序,此時新增或修改數(shù)據(jù)庫中的數(shù)據(jù),我們就能從客戶端中監(jiān)聽到

不過我建議監(jiān)聽的信息放到消息隊列中,在空閑的時候去處理,所以直接配置Canal整合RabbitMQ更好。

Canal整合RabbitMQ

修改canal.properties中的serverMode:

canal.serverMode = rabbitMQ

修改instance.properties中的topic:

canal.mq.topic=canal-routing-key

然后找到關(guān)于RabbitMQ的配置:

##################################################
#########         RabbitMQ       #############
##################################################
# 連接rabbit,寫IP,因為同個網(wǎng)絡(luò)下,所以可以寫容器名
rabbitmq.host = myrabbit
rabbitmq.virtual.host = /
# 交換器名稱,等等我們要去手動創(chuàng)建
rabbitmq.exchange = canal-exchange
# 賬密
rabbitmq.username = admin
rabbitmq.password = 123456
# 暫不支持指定端口,使用的是默認的5762,好在在本示例中適用

重新啟動容器,進入RabbitMQ管理頁面創(chuàng)建exchange交換器和隊列queue:

  • 新建exchange,命名為:canal-exchange
  • 新建queue,命名為:canal-queue
  • 綁定exchange和queue,routing-key設(shè)置為:canal-routing-key,這里對應(yīng)上面instance.properties的canal.mq.topic

順帶一提,上面這段可以忽略,因為在SpringBoot的RabbitMQ配置中,會自動創(chuàng)建交換器exchange和隊列queue,不過手動創(chuàng)建的話,可以在忽略SpringBoot的基礎(chǔ)上,直接在RabbitMQ的管理頁面上看到修改記錄的消息。

SpringBoot整合RabbitMQ

依賴:

<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>

application.yml:

spring:
  rabbitmq:
    #    host: myserverhost
    host: 192.168.0.108
    port: 5672
    username: admin
    password: RabbitMQ密碼
    # 消息確認配置項
    # 確認消息已發(fā)送到交換機(Exchange)
    publisher-confirm-type: correlated
    # 確認消息已發(fā)送到隊列(Queue)
    publisher-returns: true

RabbitMQ配置類:

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

        return template;
    }

    /**
     * template.setMessageConverter(new Jackson2JsonMessageConverter());
     * 這段和上面這行代碼解決RabbitListener循環(huán)報錯的問題
     */
    @Bean
    public SimpleRabbitListenerContainerFactory rabbitListenerContainerFactory(ConnectionFactory connectionFactory) {
        SimpleRabbitListenerContainerFactory factory = new SimpleRabbitListenerContainerFactory();
        factory.setConnectionFactory(connectionFactory);
        factory.setMessageConverter(new Jackson2JsonMessageConverter());
        return factory;
    }
}

Canal消息生產(chǎn)者:

public static final String CanalQueue = "canal-queue";
public static final String CanalExchange = "canal-exchange";
public static final String CanalRouting = "canal-routing-key";
復(fù)制代碼
/**
 * Canal消息提供者,canal-server生產(chǎn)的消息通過RabbitMQ消息隊列發(fā)送
 */
@Configuration
public class CanalProvider {
    /**
     * 隊列
     */
    @Bean
    public Queue canalQueue() {
        /**
         * durable:是否持久化,默認false,持久化隊列:會被存儲在磁盤上,當消息代理重啟時仍然存在;暫存隊列:當前連接有效
         * exclusive:默認為false,只能被當前創(chuàng)建的連接使用,而且當連接關(guān)閉后隊列即被刪除。此參考優(yōu)先級高于durable
         * autoDelete:是否自動刪除,當沒有生產(chǎn)者或者消費者使用此隊列,該隊列會自動刪除
         */
        return new Queue(RabbitConstant.CanalQueue, true);
    }

    /**
     * 交換機,這里使用直連交換機
     */
    @Bean
    DirectExchange canalExchange() {
        return new DirectExchange(RabbitConstant.CanalExchange, true, false);
    }

    /**
     * 綁定交換機和隊列,并設(shè)置匹配鍵
     */
    @Bean
    Binding bindingCanal() {
        return BindingBuilder.bind(canalQueue()).to(canalExchange()).with(RabbitConstant.CanalRouting);
    }
}

Canal消息消費者:

/**
 * 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;
        }

        // TiCDC的id,應(yīng)該具有唯一性,先保存再說
        int tid = (int) msg.get("id");
        // TiCDC生成該消息的時間戳,13位毫秒級
        long ts = (long) msg.get("ts");
        // 數(shù)據(jù)庫
        String database = (String) msg.get("database");
        // 表
        String table = (String) msg.get("table");
        // 類型:INSERT/UPDATE/DELETE
        String type = (String) msg.get("type");
        // 每一列的數(shù)據(jù)值
        List<?> data = (List<?>) msg.get("data");
        // 僅當type為UPDATE時才有值,記錄每一列的名字和UPDATE之前的數(shù)據(jù)值
        List<?> old = (List<?>) msg.get("old");

        // 跳過sys_backup,防止無限循環(huán)
        if ("sys_backup".equalsIgnoreCase(table)) {
            return;
        }

        // 只處理指定類型
        if (!"INSERT".equalsIgnoreCase(type)
                && !"UPDATE".equalsIgnoreCase(type)
                && !"DELETE".equalsIgnoreCase(type)) {
            return;
        }
    }
}

測試一下,修改MySQL中的一條消息,Canal就會發(fā)送信息到RabbitMQ,我們就能從監(jiān)聽的RabbitMQ隊列中得到該條消息。

責任編輯:武曉燕 來源: JAVA日知錄
相關(guān)推薦

2024-12-24 08:44:55

ActiveMQRabbitMQ交換機

2024-09-05 08:58:37

2024-11-04 08:02:23

SpringRabbitMQ中間件

2025-04-29 08:36:28

SpringCanal數(shù)據(jù)庫

2023-08-08 08:28:03

消息消費端Spring

2023-05-31 08:56:24

2023-03-06 08:16:04

SpringRabbitMQ

2023-08-10 11:39:54

RabbitMQSpring交換機

2020-09-08 07:37:44

springBoot MQ rabbitMQ

2023-09-26 08:11:22

Spring配置MySQL

2022-04-28 07:31:41

Springkafka數(shù)據(jù)量

2021-12-27 09:59:57

SpringCanal 中間件

2023-08-09 08:01:00

WebSockett服務(wù)器web

2022-05-30 07:31:38

SpringBoot搜索技巧

2023-06-07 08:08:37

MybatisSpringBoot

2021-08-17 06:48:43

SpringbootKafkaStream

2021-04-07 08:43:09

SpringBootRocketMQ開發(fā)技術(shù)

2021-11-15 14:02:27

RPCSpringBootRabbitMQ

2023-10-09 07:37:01

2025-06-03 02:10:00

SpringInfluxDB數(shù)據(jù)
點贊
收藏

51CTO技術(shù)棧公眾號

亚洲天堂中文在线| 一区二区黄色片| 久久最新免费视频| 久久久久久久久久一级| 999久久久亚洲| 精品日韩欧美一区二区| 国产精品免费成人| av免费在线免费观看| 波多野结衣中文一区| 国产精品香蕉国产| 亚洲一区 视频| 国产精品国内免费一区二区三区| 亚洲国产精品va在线| 538任你躁在线精品免费| 暧暧视频在线免费观看| 中文字幕在线观看一区| 久久久精彩视频| 国产sm主人调教女m视频| 亚洲一区日韩在线| 欧美成人中文字幕| 五月婷六月丁香| 亚洲资源网站| 亚洲成人网久久久| 亚洲色图欧美自拍| 97精品国产综合久久久动漫日韩| 亚洲午夜久久久久中文字幕久| 亚洲欧美日韩精品在线| 午夜小视频在线播放| 久久er99精品| 国产精品久久久久久久久久99| 精品在线视频免费| 欧美日韩视频| 久久亚洲私人国产精品va| 日本乱子伦xxxx| 尤物tv在线精品| 日韩av在线电影网| 无码人妻一区二区三区精品视频| 欧美天堂一区二区| 欧美怡红院视频| 免费激情视频在线观看| 伊人久久综合一区二区| 精品国产1区2区| 成人黄色av片| 黄色污网站在线观看| 依依成人精品视频| 9色视频在线观看| 国产淫片在线观看| 91精彩在线视频| 国产精品老牛| 国产91精品高潮白浆喷水| 国产一国产二国产三| 欧美人与禽猛交乱配视频| 久久中国妇女中文字幕| 五月天av网站| 欧美日韩91| 欧美黑人xxxx| 日本三级黄色大片| 1024成人| 日本成人在线视频网址| 亚洲成人av影片| 日韩成人dvd| 成人精品视频99在线观看免费| 在线观看免费视频a| 激情综合一区二区三区| 91免费在线观看网站| 超碰在线播放97| 成人av在线播放网址| 久久大片网站| 国产一区精品| 中文字幕日韩一区| 黑人巨茎大战欧美白妇 | 亚洲理论在线a中文字幕| 自拍偷拍视频亚洲| 欧美激情理论| 欧美韩日一区二区| 国产69精品久久久久久久久久| 亚洲专区欧美专区| 国产精品美乳在线观看| 国产乱淫a∨片免费观看| 国产精品中文字幕一区二区三区| 99久久综合狠狠综合久久止| 午夜av免费观看| 国产精品天美传媒沈樵| 国产香蕉一区二区三区| 国产精品偷拍| 欧美三级在线看| 精品国产一二区| 国产一区二区电影在线观看| 久久精品国产91精品亚洲| 精品99久久久久成人网站免费 | 91热这里只有精品| 日韩08精品| 精品亚洲一区二区三区四区五区 | 亚洲精品久久嫩草网站秘色| 免费 成 人 黄 色| 精品福利在线| 亚洲高清一区二| 亚洲一区电影在线观看| 亚洲精品免费观看| 国产有码在线一区二区视频| 手机看片国产1024| 中文字幕亚洲欧美在线不卡| 日韩少妇内射免费播放| 色成人综合网| 亚洲欧美第一页| 久久精品99久久久久久| 日本在线不卡视频一二三区| 国产精品二区三区| 2019中文字幕在线视频| 黄网动漫久久久| 欧美日韩理论片| 精品一区二区三区中文字幕老牛| 久久性色av| 久久久成人av| 波多野结衣一二区| 99精品国产91久久久久久| 国产精品h视频| 姬川优奈av一区二区在线电影| 日韩一区二区视频在线观看| 国产一区二区三区精品在线| 99xxxx成人网| 成人在线观看av| 里番在线观看网站| 色婷婷国产精品| 99久久免费看精品国产一区| 中出一区二区| 成人免费网站在线观看| 每日更新在线观看av| 午夜精品国产更新| 日本女人性视频| 亚洲精品成人无限看| 国产精品久久久久免费a∨大胸| 岛国大片在线播放| av中文在线| 欧美色视频日本版| 精品熟女一区二区三区| 欧美日韩一区二区国产| 91理论片午午论夜理片久久| av男人的天堂在线| 欧美色图天堂网| 精品一区二区三孕妇视频| 美女久久一区| 日本高清一区| 一区二区视频免费完整版观看| 国产视频亚洲精品| 天堂а√在线中文在线新版| 不卡区在线中文字幕| www.av蜜桃| 激情亚洲另类图片区小说区| 久久久久日韩精品久久久男男| 国产chinasex对白videos麻豆| 亚洲免费伊人电影| 男插女视频网站| 亚洲午夜极品| 国产在线欧美日韩| 黄色漫画在线免费看| 日韩av综合中文字幕| 国产精品美女久久久久av爽| 久久综合中文字幕| jizz欧美激情18| 久久香蕉国产| 91亚洲精品一区| 日本高清成人vr专区| 日韩久久久久久| 五月婷婷激情网| 久久精品一区八戒影视| 久久人妻精品白浆国产| 日韩一区三区| 亚洲综合社区网| av免费不卡| 亚洲欧美制服第一页| 国产精品欧美综合| 亚洲欧美日韩在线播放| 五月天丁香社区| 亚洲一区二区伦理| 亚洲国产精品综合| 日本少妇精品亚洲第一区| 91精品国产91久久久久久最新| 日本大臀精品| 91精品国产欧美日韩| 国产精品99精品无码视| 久久久国际精品| 亚洲高清视频免费| 亚洲作爱视频| 一区二区在线不卡| 久9re热视频这里只有精品| 日韩暖暖在线视频| 性欧美高清come| 亚洲男人的天堂网站| 国产一区二区三区四区视频 | 懂色av一区二区在线播放| 国产中文字幕乱人伦在线观看| 亚洲va久久| 3d蒂法精品啪啪一区二区免费| 国产激情在线播放| 神马久久久久久| 天天干天天色天天| 欧美精品久久久久久久久老牛影院| 国产一级性生活| 国产精品九色蝌蚪自拍| a级在线观看视频| 国产精品一区二区在线播放| 日韩a在线播放| 欧美精品入口| 日韩一区二区三区高清| 高清日韩中文字幕| 国产在线观看精品| 亚洲天堂资源| 久久久久久999| 韩国av网站在线| 国产一区二区三区高清在线观看| 亚洲av无码乱码国产精品久久| 在线中文字幕一区| 日本熟伦人妇xxxx| 亚洲男人天堂av| 懂色av蜜桃av| 久久久久久久综合色一本| 亚洲成a人片在线www| 精品一区二区三区在线观看| 日韩亚洲在线视频| 亚洲精品婷婷| 国产女教师bbwbbwbbw| 99久久婷婷这里只有精品 | 欧美大人香蕉在线| 欧美日韩一区二区三区在线视频| 亚洲国产欧美国产第一区| 国产欧美精品va在线观看| 新片速递亚洲合集欧美合集| 69久久夜色精品国产7777| 黄网av在线| 欧美激情精品久久久久久免费印度 | 麻豆中文字幕在线观看| 超碰成人久久| 视频一区视频二区视频| 国产亚洲电影| 日本在线视频不卡| 国产99久久| 欧美人与物videos另类| 色综合久久中文| 久久99精品久久久久子伦| 成人高潮视频| 国产66精品久久久久999小说| 免费精品一区二区三区在线观看| 久久综合综合久久综合| 九九九九九精品| 欧美人妖在线观看| 另类视频在线观看+1080p| 日韩极品在线| 欧美精品久久久| 国产一区二区区别| 水蜜桃一区二区| 欧美xxav| 中文字幕制服丝袜在线| 91精品国产91久久综合| 在线观看三级网站| 欧美午夜电影在线观看 | 涩涩日韩在线| 日韩在线视屏| 久操手机在线视频| 亚洲国产专区校园欧美| 成人免费播放器| 性感少妇一区| 国产wwwxx| 国内外成人在线| 亚洲精品第二页| 国产亚洲一本大道中文在线| 国产激情av在线| 亚洲视频一区二区免费在线观看| 国产精品视频一区二区三| 亚洲一区二区美女| 91video| 欧美日韩夫妻久久| 性色av蜜臀av| 亚洲欧美色婷婷| 免费高清完整在线观看| 欧美日韩福利电影| 2022成人影院| 成人福利网站在线观看| 第一区第二区在线| 日韩理论片在线观看| 欧美一区二区三区另类| 可以在线看的av网站| 日韩国产在线一| 美女流白浆视频| 久久九九影视网| 亚洲综合网在线| 欧美色视频日本高清在线观看| 一级特黄aaaaaa大片| 精品国产一区二区三区忘忧草 | 日韩在线观看网址| av中文在线资源| 国产精品美腿一区在线看| 91蝌蚪精品视频| 午夜精品视频在线观看一区二区| 欧美一区亚洲| 少妇人妻互换不带套| 国产成人在线电影| 中文字幕av久久爽一区| 亚洲国产日韩a在线播放| 艳妇乳肉豪妇荡乳av无码福利| 欧美r级在线观看| av在线天堂| 性欧美视频videos6一9| 虎白女粉嫩尤物福利视频| 久久综合影视| 蜜桃色一区二区三区| 国产精品毛片大码女人| 亚洲伊人成人网| 欧美mv日韩mv亚洲| 黄av在线播放| 国产精品吴梦梦| 日韩精品导航| 福利在线一区二区| 精品亚洲成a人在线观看| 日本黄色网址大全| 亚洲国产视频直播| 99草在线视频| www.欧美三级电影.com| 偷拍视频一区二区三区| 国产乱子伦精品| 欧美一区二区三区久久精品茉莉花| 五月婷婷激情久久| 久久一区二区三区四区| 国产精品9191| 精品国产乱码久久久久久久| 免费在线看黄| 国产精自产拍久久久久久蜜| 精品日韩免费| 国产性生交xxxxx免费| 2021中文字幕一区亚洲| 国产精品成人网站| 欧美成人精品二区三区99精品| 欧美a在线看| 国产精品自拍网| 91视频综合| 一区二区三区 日韩| 国产精品丝袜一区| 波多野结衣不卡| 国产亚洲人成网站在线观看| 中文字幕在线视频网站| 蜜桃导航-精品导航| 一级成人国产| 老鸭窝一区二区| 欧美性xxxxx极品娇小| 日色在线视频| 日韩免费观看网站| 成人羞羞网站入口| 久热精品在线播放| 亚洲欧洲精品一区二区精品久久久| 超碰在线免费97| 久久精品国产精品亚洲| 亚洲福利影视| 青青草免费在线视频观看| 国产成人综合亚洲91猫咪| 久久午夜鲁丝片午夜精品| 欧美va天堂va视频va在线| а_天堂中文在线| 麻豆成人av| 日本欧美一区二区三区| 国产jizz18女人高潮| 欧美一二三四区在线| 色呦呦视频在线观看| 久久精品日产第一区二区三区精品版 | 成人做爰www免费看视频网站| 亚洲成人tv| 日本一级大毛片a一 | 91青青草视频| 日韩在线观看免费全| 国产区一区二| 阿v天堂2017| 国产精品人成在线观看免费| 国产绳艺sm调教室论坛| 国内精品国产三级国产在线专 | 久久视频免费在线| 不卡的av在线播放| 波多野结衣视频观看| 久热国产精品视频| 欧美电影完整版在线观看| av免费网站观看| 亚洲乱码中文字幕综合| 四虎永久在线观看| 国产精品亚发布| 在线播放精品| 国产馆在线观看| 亚洲а∨天堂久久精品9966| 欧美日韩成人影院| 欧美一级黄色录像片| 91蜜桃免费观看视频| 97人妻精品一区二区三区视频| 久久久人成影片一区二区三区观看| 国产欧美日韩影院| 99999精品| 色诱亚洲精品久久久久久| 最新黄网在线观看| 鲁片一区二区三区| 蜜臀av一级做a爰片久久| 国产一级二级三级视频| 一区二区三区精品99久久| swag国产精品一区二区| 亚洲色图38p|