【canal】canal同步msyql到redis

java层面 canal client配置pom文件

复制代码
     <dependency>
            <groupId>com.google.code.gson</groupId>
            <artifactId>gson</artifactId>
            <version>2.13.2</version>
        </dependency>
        <dependency>
            <groupId>com.google.guava</groupId>
            <artifactId>guava</artifactId>
            <version>33.5.0-jre</version>
        </dependency>
        <dependency>
            <groupId>com.alibaba.otter</groupId>
            <artifactId>canal.client</artifactId>
            <version>1.1.7</version> <!-- 使用最新版本 -->
        </dependency>
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>5.1.47</version>
        </dependency>

        <!-- Redisson -->
        <dependency>
            <groupId>org.redisson</groupId>
            <artifactId>redisson</artifactId>
            <version>3.23.5</version>
        </dependency>

        <dependency>
            <groupId>com.alibaba.otter</groupId>
            <artifactId>canal.protocol</artifactId>
            <version>1.1.7</version>
        </dependency>

java client

1.将接收到的canal数据2进制转为10进制数据

2.redis client需要2进制转10进制,否则key是10进制,内部还是2进制

java 复制代码
package org.example.cannal.redis;

import com.alibaba.otter.canal.client.CanalConnectors;
import com.alibaba.otter.canal.client.CanalConnector;
import com.alibaba.otter.canal.protocol.Message;
import com.alibaba.otter.canal.protocol.CanalEntry.*;
import org.redisson.api.RMap;
import org.redisson.api.RedissonClient;
import org.redisson.config.Config;

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

public class CanalBinaryParseWithRedisson {

    public static void main(String[] args) {
        // 1. 创建 Canal 连接
        CanalConnector connector = CanalConnectors.newSingleConnector(
                new InetSocketAddress("127.0.0.1", 11111),
                "example", "", ""
        );

        // 2. 创建 Redisson 客户端
        RedissonClient redisson = createRedissonClient();

        try {
            connector.connect();
            connector.subscribe(".*\\..*");
            connector.rollback();

            while (true) {
                Message message = connector.getWithoutAck(100); // 每次取100条
                long batchId = message.getId();
                int size = message.getEntries().size();

                if (batchId == -1 || size == 0) {
                    try {
                        Thread.sleep(1000);
                    } catch (InterruptedException ignored) {}
                } else {
                    handleEntries(message.getEntries(), redisson);
                    connector.ack(batchId); // 确认消费
                }
            }
        } finally {
            connector.disconnect();
            redisson.shutdown();
        }
    }

    private static void handleEntries(List<Entry> entrys, RedissonClient redisson) {
        for (Entry entry : entrys) {
            if (entry.getEntryType() == EntryType.TRANSACTIONBEGIN ||
                entry.getEntryType() == EntryType.TRANSACTIONEND) {
                continue;
            }

            RowChange rowChange;
            try {
                // 关键:把二进制解析成 RowChange
                rowChange = RowChange.parseFrom(entry.getStoreValue());
            } catch (Exception e) {
                throw new RuntimeException("解析 RowChange 失败", e);
            }

            EventType eventType = rowChange.getEventType();
            String tableName = entry.getHeader().getTableName();

            for (RowData rowData : rowChange.getRowDatasList()) {
                if (eventType == EventType.INSERT || eventType == EventType.UPDATE) {
                    Map<String, String> dataMap = new HashMap<>();
                    String id = null;
                    for (Column column : rowData.getAfterColumnsList()) {
                        dataMap.put(column.getName(), column.getValue());
                        if ("id".equals(column.getName())) {
                            id = column.getValue();
                        }
                    }
                    // 用 Redisson 写入 Redis Hash
                    RMap<String, String> map = redisson.getMap(tableName + ":" + id);
                    map.putAll(dataMap);
                    System.out.println("写入 Redis: " + tableName + ":" + id + " -> " + dataMap);
                } else if (eventType == EventType.DELETE) {
                    String id = null;
                    for (Column column : rowData.getBeforeColumnsList()) {
                        if ("id".equals(column.getName())) {
                            id = column.getValue();
                            break;
                        }
                    }
                    // 删除 Redis Key
                    redisson.getMap(tableName + ":" + id).delete();
                    System.out.println("删除 Redis: " + tableName + ":" + id);
                }
            }
        }
    }

    private static RedissonClient createRedissonClient() {
        Config config = new Config();
        // 1. 配置Redis连接(单节点为例)
        config.useSingleServer()
                .setAddress("redis://127.0.0.1:6379")
//                .setPassword("你的密码") // 无密码则省略
                .setDatabase(1);

        // 2. 关键:设置序列化器为StringCodec(明文字符串序列化)
        config.setCodec(new org.redisson.client.codec.StringCodec());

        return org.redisson.Redisson.create(config);
    }
}
相关推荐
Rabbit_QL17 小时前
【水印添加工具】从零设计一个工程级 Python 图片水印工具:WaterMask 架构与实现
开发语言·python
预立科技18 小时前
Redis 中 Lua 与 Pipeline 的相同点,区别,使用场景
redis·pipeline·lua
曲幽19 小时前
FastAPI多进程部署:定时任务重复执行?手把手教你用锁搞定
redis·python·fastapi·web·lock·works
森屿~~19 小时前
AI 手势识别系统:踩坑与实现全记录 (PyTorch + MediaPipe)
人工智能·pytorch·python
忧郁的橙子.20 小时前
26期_01_Pyhton文件的操作
开发语言·python
wWYy.20 小时前
详解redis(15):缓存雪崩
数据库·redis·缓存
这周也會开心21 小时前
Redis相关知识点
数据库·redis·缓存
Whisper_Sy21 小时前
Flutter for OpenHarmony移动数据使用监管助手App实战 - 应用列表实现
android·开发语言·javascript·flutter·php
小CC吃豆子21 小时前
Python爬虫
开发语言·python
June bug21 小时前
(#字符串处理)字符串中第一个不重复的字母
python·leetcode·面试·职场和发展·跳槽