Java 监听Mysql binlog

这篇具有很好参考价值的文章主要介绍了Java 监听Mysql binlog。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

使用 mysql-binlog-connector-java

1. mysql-binlog-connector-java 官网
2. Java代码中,如何监控Mysql的binlog?

前置条件
1. mysql服务器表结构
CREATE TABLE `student` (
  `id` int NOT NULL AUTO_INCREMENT,
  `name` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL,
  `age` int NOT NULL,
  `code` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL,
  PRIMARY KEY (`id`),
  KEY `idx_name` (`name`) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=19 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_general_ci;
2. 开启master的mysql 服务器的log_bin
show variables like 'log_bin';

Java 监听Mysql binlog,java,mysql
如果没有,那么设置文件中增加配置

log_bin=mysql-bin
binlog-format=ROW
server-id=1

重启服务

  • 在配置文件中加入了log_bin配置项后,表示启用了binlog
  • binlog-format是binlog的日志格式,支持三种类型,分别是STATEMENT、ROW、MIXED,我们在这里使用ROW模式
  • server-id用于标识一个sql语句是从哪一个server写入的,这里一定要进行设置,否则我们在后面的代码中会无法正常监听到事件
引入maven依赖
<dependency>
    <groupId>com.zendesk</groupId>
    <artifactId>mysql-binlog-connector-java</artifactId>
    <version>0.25.0</version>
</dependency>
代码块实现监听

查看一个简单的case

public static void main(String[] args) {
		// 这里的账号必须要有权限访问
        BinaryLogClient client = new BinaryLogClient("127.0.0.1", 3306, "root", "root");
        // 反序列化配置
        EventDeserializer eventDeserializer = new EventDeserializer();
        eventDeserializer.setCompatibilityMode(EventDeserializer.CompatibilityMode.DATE_AND_TIME_AS_LONG
//                EventDeserializer.CompatibilityMode.CHAR_AND_BINARY_AS_BYTE_ARRAY
        );
        // 设置反序列化配置
        client.setEventDeserializer(eventDeserializer);
        // 设置自己的client作为服务器的id
        client.setServerId(3);
        // 可选,设置start fileName+position
//        client.setBinlogFilename("master-bin.000080");
//        client.setBinlogPosition(219);

        client.registerEventListener(event -> {
            EventData data = event.getData();
            String tableName;
            if (data instanceof TableMapEventData) {
                System.out.println("Table:");
                TableMapEventData tableMapEventData = (TableMapEventData) data;
                System.out.println(tableMapEventData.getTableId() + ": [" + tableMapEventData.getDatabase() + "." + tableMapEventData.getTable() + "]");
                tableName = tableMapEventData.getTable();
                // 如果是不处理的表,那么返回
                if (!Objects.equals(tableName, "student"))
                    return;
            }
            if (data instanceof UpdateRowsEventData) {
//                System.out.println("Update:");
//                System.out.println(data);
                // 获取对应的操作对象的json化数据
                UpdateRowsEventData udata = (UpdateRowsEventData) data;
                List<Map.Entry<Serializable[], Serializable[]>> rows = udata.getRows();
                for (Map.Entry<Serializable[], Serializable[]> row : rows) {
                    List<Serializable> entries = Arrays.asList(row.getValue());
                    JSONObject dataObject = getDataObject(entries);
                    System.out.println(dataObject);
                }
            } else if (data instanceof WriteRowsEventData) {
                WriteRowsEventData wData = new WriteRowsEventData();
                wData.getIncludedColumns();
                wData.getRows();

                System.out.println("Insert:");
                System.out.println(data);
            } else if (data instanceof DeleteRowsEventData) {
                System.out.println("Delete:");
                System.out.println(data);
            }
        });
        try {
            client.connect();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }


    /**
     * 根据message获取对象
     */
    private static JSONObject getDataObject(List<Serializable> message) {
        JSONObject resultObject = new JSONObject();
        String format = "{\"id\":\"0\",\"name\":\"1\",\"age\":\"2\",\"code\":\"3\"}";
        JSONObject json = JSON.parseObject(format);
        for (String key : json.keySet()) {
            resultObject.put(key, message.get(json.getInteger(key)));
        }
        return resultObject;
    }

首先,创建一个BinaryLogClient客户端对象,初始化时需要传入mysql的连接信息,创建完成后,给客户端注册一个监听器,来实现它对binlog的监听和解析。在监听器中,我们暂时只对4种类型的事件数据进行了处理,除了WriteRowsEventData、DeleteRowsEventData、UpdateRowsEventData对应增删改操作类型的事件数据外,还有一个TableMapEventData类型的数据,包含了表的对应关系,在后面的例子中再具体说明。
在这里,客户端监听到的是数据库级别的所有事件,并且可以监听到表的DML语句和DDL语句,所以我们只需要处理我们关心的事件数据就行,否则会收到大量的冗余数据。

启动程序,控制台输出:
Java 监听Mysql binlog,java,mysql

我们需要知道表的信息或者的行信息,在反序列化的过程中,我们调用UpdateRowsEventData中只有行的index,而没有name,那么这个时候需要我们自己定义每列对应的续好了,然后解析获取到的row的信息。
Java 监听Mysql binlog,java,mysql
方法如:

private static JSONObject getDataObject(List<Serializable> message) {
        JSONObject resultObject = new JSONObject();
        String format = "{\"id\":\"0\",\"name\":\"1\",\"age\":\"2\",\"code\":\"3\"}";
        JSONObject json = JSON.parseObject(format);
        for (String key : json.keySet()) {
            resultObject.put(key, message.get(json.getInteger(key)));
        }
        return resultObject;
    }
表和列信息模板

在getDataObject中,我们需要知道每个cloumnId对应的name,那么,我们需要自定义一个format,但是columnId和index的关系我们可以从information_schema中获取,当然,这就需要处理了,这里另说。
Java 监听Mysql binlog,java,mysql

指定binlog的起始位置

By default, BinaryLogClient starts from the current (at the time of connect) master binlog position. If you wish to kick off from a specific filename or position, use client.setBinlogFilename(filename) + client.setBinlogPosition(position).

通常来说,监听从启动开始,但是也可以指定fileName+position

// 可选,设置start fileName+position,后续的话会都会开始执行的,包括后续的binlog file
client.setBinlogFilename("master-bin.000079");
client.setBinlogPosition(4);

查找binLog中的信息

参考:MySQL Binlog二进制日志基于position位置点恢复数据

如果无法确定启动的bin文件和postion,可以查看相关的binlog 文件,

查找binlog的文件名

查看master-server的配置文件,查看目录和文件前缀 mysql.ini
Java 监听Mysql binlog,java,mysql
在目录下查找对应的binLog文件

Java 监听Mysql binlog,java,mysql

查看binlong文件中的信息
show binlog events in 'master-bin.000080';

Java 监听Mysql binlog,java,mysql

总结

1. git代码文章来源地址https://www.toymoban.com/news/detail-629483.html

到了这里,关于Java 监听Mysql binlog的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • MySQL 开启配置binlog以及通过binlog恢复数据

       binlog是MySQL sever层维护的一种二进制日志,binlog是记录所有数据库表结构变更(例如CREATE、ALTER TABLE、DROP等)以及表数据修改(INSERT、UPDATE、DELETE、TRUNCATE等)的二进制日志。不会记录SELECT和SHOW这类操作,因为这类操作对数据本身并没有修改。 作用主要有: 主从复制:在

    2024年02月03日
    浏览(38)
  • mysql binlog 回滚

    mysqlbinlog 严格来说mysqlbinlog 不能算回滚,他只是将过去的数据修改记录 重新执行一遍,但是从结果上来看,他也算把数据恢复到任意时间点了,举例来说在昨天的某一刻误删除了一条数据,导致其他数据存储都是异常,今天才发现,现在我希望回滚到那一刻,那么我只要在

    2024年02月12日
    浏览(34)
  • mysql的binlog日志

    一、查看和配置binlog 1、log_bin 是否开启binlog,指定日志文件路径 2、log_bin_basename 和 log_bin_index 日志文件基础名和索引名(*好像不能用来设置只是展示作用,我设置时时会报错无法启动服务) 3、binlog_format 日志格式 4、binlog_error_action 设置当binlog日志数据一致性遭到破坏或者复

    2024年02月16日
    浏览(27)
  • MySQL 如何查询binlog

    binlog开启成功之后,binlog文件的位置可以在my.inf配置文件中查看。也可以在MySQL的命令行中查看。命令行查看代码如下: 然后可以看到MySQL的binlog相关信息:  然后进入相关目录下  因为此文件为字节码文件,直接查看是不可读的,因此需要借助MySQL 的mysqlbinlog命令: 然后就

    2024年02月13日
    浏览(27)
  • 【mysql】binlog日志

    1.1 基本说明 1.全称binary log,二进制日志 2.记录了所有的DDL语句(Data Definition Language数据定义语言)和DML语句(Data Manipulation /məˌnɪpjuˈleɪʃn/ Language数据操作语言) 3.不包括数据查询语句(select、show) 4.作用:灾难时的数据恢复;mysql的主从复制 5.mysql8.0版本,默认二进制日

    2024年02月13日
    浏览(21)
  • mysql binlog

    二进制日志文件记录了数据库修改的事件,像表的修改,表数据的变更等。也包含潜在的可能修改数据的语句事件。如一些delete或update最后修改的数据行可能是0,也会被记录在binlog中(和日志格式也有一定关系,非row-based)。除此之外binlog还会记录语句的执行时间信息。 binlo

    2024年02月22日
    浏览(24)
  • 清理MySQL中的binlog

    Mysql的binlog开启后一直没清理,占用太大空间 expire_logs_days=0: 这里的值如果为0,表示所有binlog日志永久都不会失效,不会自动删除; 这里的值如果为30,表示只保留最近30天。 永久生效(重启后即生效) 修改配置文件my.cnf文件: vim /etc/my.cnf 如果binlog非常多,推荐使用purge命令

    2023年04月08日
    浏览(26)
  • 开启MySQL的binlog日志

    查看MySQL的binlog模式   几个关于binlog常用的命令   binlog 就是binary log,二进制日志文件,这个文件记录了mysql所有的dml操作。通过binlog日志我们可以做数据恢复,做主住复制和主从复制等等。对于开发者可能对binlog并不怎么关注,但是对于运维或者架构人员来讲是非常重要的

    2024年01月21日
    浏览(35)
  • MySQL:binlog启动与查看

    Mysql binlog,即二进制日志,是MySQL最重要的日志,它记录了所有的DDL和DML语句(除了数据查询语句select、show等),以数据形式记录,还包含语句执行所消耗的时间。 binlog的主要目的是复制和恢复。 如何查看MySQL是否开启了binlog? 登录MySQL后,输入: 显示off则未开启,显示o

    2024年02月12日
    浏览(22)
  • Mysql之binlog日志浅析

      Binlog是MySQL数据库中的二进制日志,用于记录数据库中所有修改操作,包括增删改等操作。binlog以二进制格式保存,可以通过解析binlog文件来查看数据库的操作历史记录。binlog日志可以用于数据恢复、数据备份、数据同步等场景。在MySQL数据库中,binlog有两种模式:state

    2024年02月02日
    浏览(36)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包