缓存同步的常见方式

  • 设置有效期
    • 给缓存社会有效期,到期删除,再来查询,就查数据库,在放入缓存
  • 同步双写
    • 修改DB时,同步修改DB
  • 异步通知
    • 修改DB时,发送事件通知
    • 比如:MQ、Canal

上面场景,只要不傻都知道。各有优势,我就不说了,本文介绍Canal

什么是Canal 需要JDK

Canal主要用途是基于 MySQL 数据库增量日志解析,提供增量数据订阅和消费

Cnanl地址:https://github.com/alibaba/canal

Canal原理

利用Mysql主从同步机制来实现的。

  • MySQL master 将数据变更写入二进制日志( binary log, 其中记录叫做二进制日志事件binary log events,可以通过 show binlog events 进行查看)
  • MySQL slave 将 master 的 binary log events 拷贝到它的中继日志(relay log)
  • MySQL slave 重放 relay log 中事件,将数据变更反映它自己的数据

canal 工作原理

  • canal 模拟 MySQL slave 的交互协议,伪装自己为 MySQL slave ,向 MySQL master 发送dump 协议
  • MySQL master 收到 dump 请求,开始推送 binary log 给 slave (即 canal )
  • canal 解析 binary log 对象(原始为 byte 流)

Canal 伪装成slave节点,监听master binary log的变化。将此信息发送到Canal客户端中。就完成了异步通知了!

修改Mysql配置

修改mysql配置,配置文件一般就是my.cnf中,不管是啥,只要是[mysqld]就行了

[mysqld]
# 这是指定服务编号,我怕你重启报错,建议加上吧!
server-id= 1
# 指定binlog日志名称,傻逼mysql有些文件夹不允许随便指定,error.log说了没文件,创建了文件也不行,去他妈的!
log-bin=/var/log/mysql/mysql-bin.log
# 指定mysql的binlog日志记录哪个db
binlog-do-db=hello,mysql
binlog-format=ROW # 选择 ROW 模式

重启mysql

systemctl restart mysql

确定重启成功了

systemctl restart mysql

给Canal创建一个配置用户,用于管理,最好限制操作库的权限 参考:https://www.zanglikun.com/4473.html

CREATE USER 'canal'@'%' IDENTIFIED BY 'canal';
GRANT all privileges ON *.* TO 'canal'@'%';
-- 如果你root没有系统用户则不可用使用授权命令 添加身份:grant system_user on *.* to 'root';
-- 不知道是不是有啥问题,反正得要配置一下这个
update `mysql`.`user` SET `plugin` ='mysql_native_password' where User = 'canal';
flush privileges;


Drop User 'canal'@'%';
select Host,User from mysql.user;

查看主从状态

-- 查看主从状态
SHOW MASTER STATUS;

安装Canal

下载:https://github.com/alibaba/canal/releases/tag/canal-1.1.6

建议看官方教程:https://github.com/alibaba/canal/wiki/QuickStart

添加一些配置文件

conf/canal.properties

# tcp bind ip
canal.ip =192.168.31.19

example/instance.properties

## mysql serverId , v1.0.26+ will autoGen 这里说了1.0.26 就不用配置了
# canal.instance.mysql.slaveId=0

# position info
canal.instance.master.address=127.0.0.1:3306
# 配置主从在用的binlog文件名 由执行show master status;得到
canal.instance.master.journal.name=mysql-bin.000006
canal.instance.master.position=157
canal.instance.master.timestamp=
canal.instance.master.gtid=

# username/password 我们刚才创建的就是canal,密码也是
canal.instance.dbUsername=canal
canal.instance.dbPassword=canal
canal.instance.connectionCharset = UTF-8

我当前是Ubuntu,安装JDK8

apt install openjdk-8-jdk

禁用防火墙

systemctl stop firewalld.service


systemctl status firewalld.service 

启动Canal

文件需要执行的权限
./startup.sh

启动完成后,看一下端口是否被占用

lsof -i:11111

如果被占用,说明启动成功!

最好看一下控制台

cat /你的Canal安装路径/logs/canal/canal.log

springboot接入 canal

本次使用是这个:https://github.com/NormanGyllenhaal/canal-client 有其他java库,自行寻找其他也可使用!

        <dependency>
            <groupId>top.javatool</groupId>
            <artifactId>canal-spring-boot-starter</artifactId>
            <version>1.2.1-RELEASE</version>
        </dependency>

配置文件

# Canal 配置信息
#  多个服务地址以逗号隔开
canal.server=192.168.31.34:11111
#  instance名称
canal.destination=example
# 消费时间间隔,单位秒,默认1
canal.timeout=3

监听User表 增、删、改。

import com.zanglikun.springdataredisdemo.pojo.User;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Component;
import top.javatool.canal.client.annotation.CanalTable;
import top.javatool.canal.client.handler.EntryHandler;

@Slf4j
@Component
@CanalTable("User") // ① 要监听到表
public class CanalTest implements EntryHandler<User> { // ② 实现接口,指定User,主动重写方法

    @Override
    public void insert(User user) {
        EntryHandler.super.insert(user);
        log.info("插入了数据"+user.toString());
    }

    @Override
    public void update(User before, User after) {
        EntryHandler.super.update(before, after);
        log.info("更新前数据"+before.toString());
        log.info("更新后数据"+after.toString());
    }

    @Override
    public void delete(User user) {
        EntryHandler.super.delete(user);
        System.out.println("执行了");
        log.info("删除了"+user.toString());
    }
}

启动即可

日志会打印出变化,我们后期根据User的变化更新相应的缓存数据即可!

特殊说明:
上述文章均是作者实际操作后产出。烦请各位,请勿直接盗用!转载记得标注原文链接:www.zanglikun.com
第三方平台不会及时更新本文最新内容。如果发现本文资料不全,可访问本人的Java博客搜索:标题关键字。以获取最新全部资料 ❤