java主从服务搭建及应用

mysql主从复制的原理
在这里插入图片描述
首先master服务需要开启binary-log日志文件,
1、master服务会开启一个log dump线程,将所有的DDL,DML,TCL写入BinaryLog的二进制日志文件中
2、slave服务会通过I/O线程请求master服务的bin-log日志文件,并写入到slave服务中的relaylog中继日志文件中
3、slave服务再通过sql线程读取relayLog日志,并解析成具体操作,来实现主从的操作一致,从而达到最终的数据一致。

一、搭建mysql主从服务
【模拟服务】:通过两个不同的端口号进行模拟两台mysql服务。
安装mysql服务。将master服务端口号定义为33337,将slave服务端口号定义为33338。

1、master - 修改mysql配置配置文件(my.ini)。
配置中有的就覆盖,配置中没有的就添加

[client]
port=33337

[mysqld]
#二进制文件
log-bin=mysql-bin
#服务id
server-id=1  //master服务和slave服务的selver-id一定不能相同
port=33337
basedir="D:/software/mysql/master"
datadir="D:/software/mysql/master/Data"
secure-file-priv="D:/software/mysql/master/Uploads"

slave - 修改mysql配置配置文件(my.ini)。
配置中有的就覆盖,配置中没有的就添加

[client]
port=33338

[mysqld]
#二进制文件
log-bin=mysql-bin
#服务id
server-id=100  //master服务和slave服务的selver-id一定不能相同
port=33338
basedir="D:/software/mysql/slave"
datadir="D:/software/mysql/slave/Data"
secure-file-priv="D:/software/mysql/slave/Uploads"

2、通过管理员身份打开打开cmd,然后分别进入master和slave的bin目录下
执行安装命令。

# master
mysqld --install master --defaults-file="D:\software\mysql\master\my.ini"

# slave
mysqld --install master --defaults-file="D:\software\mysql\slave\my.ini"

3、通过命令打开mysql服务。
net start 服务名

net start master  //打开master服务
net start slave    //打开slave服务

net stop master   //关闭master服务
net stop slave     //关闭slave服务
【注意】:每次修改完配置文件一定要重启服务
如果启动服务报错了,在安装mysql服务的data文件中找到.err文件,查看启动报错日志。如果是复制之前安装好的mysql服务,在启动的两台或多态服务的时候会报uuids相同。这时只用在data文件中找到auto.cnf文件并删除就行,然后重启服务。

4、通过连接mysql服务中连接好master服务和slave服务。分别进行服务主从服务配置

[1]:配置master服务
# 在master服务中授予slaveuser账号,密码为root,ip为192.168.10.1,查看ip的方式ipconfig
a: grant replication slave on *.* to 'slaveuser'@'192.168.10.1'identified by 'root'
# 查看master服务的状态
b: show master status


[2]:配置slave服务
# 身份认证
a: 	change master to
 	master_host='192.168.10.1',       #master IP
 	master_user='slaveuser',          #master数据库通过GRANT授权的账号
 	master_password='root',           #master数据库通过GRANT授权的密码
 	master_port=33337,                #master数据库的端口
 	master_log_file='mysql-bin.000010', #master数据库中通过show master status显示的File名称
 	master_log_pos=457               #master数据库的通过show master status显示的Position的值
# 启动slave服务
b: 	start slave
# 查看slave服务状态
c: 	show slave status
当显示为下图的状态表示为主从配置已经成功了

在这里插入图片描述
【当slave_sql_Running显示为null的时候,再次运行a步骤时,提示io线程在使用的时候的时候,可以使用 STOP SLAVE IO_THREAD 结束线程。

二、springboot中进行读写分离
1、导入pom依赖

<dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-aop</artifactId>
        </dependency>

        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
        </dependency>
        
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-jdbc</artifactId>
        </dependency>

        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
        </dependency>
 
        <dependency>
            <groupId>com.baomidou</groupId>
            <artifactId>mybatis-plus-boot-starter</artifactId>
            <version>3.0.5</version>
        </dependency>
        
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>druid</artifactId>
            <version>1.1.9</version>
        </dependency>
    </dependencies>

2、配置yml

mybatis-plus:
  mapper-locations: classpath:mapper/*Mapper.xml
  configuration:
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
    call-setters-on-nulls: true
  type-aliases-package: com.mmm.domain

spring:
  main:
    allow-bean-definition-overriding: true #bean名称相同的话,允许覆盖,后者覆盖前者
  application: aop-server
  datasource:
    master:
      type: com.alibaba.druid.pool.DruidDataSource
      driver-class-name: com.mysql.cj.jdbc.Driver
      url: jdbc:mysql://localhost:33337/aaaa?useLegacyDatetimeCode=false&serverTimezone=Asia/Shanghai&useunicode=true&characterEncoding=utf-8
      username: root
      password: root
    slave:
      type: com.alibaba.druid.pool.DruidDataSource
      driver-class-name: com.mysql.cj.jdbc.Driver
      url: jdbc:mysql://localhost:33338/aaaa?useLegacyDatetimeCode=false&serverTimezone=Asia/Shanghai&useunicode=true&characterEncoding=utf-8
      username: root
      password: root

3、写配置文件

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public class RoutingDataSourceHolder {
    private static final Logger logger = LoggerFactory.getLogger(MyRoutingDataSource.class);

    private static final ThreadLocal<String> dataSources = new ThreadLocal<>();

    public static void setDataSource(String dataSourceName){
        if(dataSources.get() == null){
            dataSources.set(dataSourceName);
            logger.info("设置数据源{}",dataSourceName);
        }
    }

    public static String getDataSource(){
        return dataSources.get();
    }

    public static void clearDataSource(){
        dataSources.remove();
    }
}
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;


public class MyRoutingDataSource extends AbstractRoutingDataSource {

    private static final Logger logger = LoggerFactory.getLogger(MyRoutingDataSource.class);

    @Override
    protected Object determineCurrentLookupKey() {
        String dataSource = RoutingDataSourceHolder.getDataSource();
        logger.info("使用的数据源{}",dataSource);
        return dataSource;
    }
}
import com.alibaba.druid.pool.DruidDataSource;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.boot.jdbc.DataSourceBuilder;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.DependsOn;
import org.springframework.context.annotation.Primary;
import org.springframework.util.StringUtils;

import javax.sql.DataSource;
import java.util.HashMap;
import java.util.Map;

@Configuration
public class MyDataSourceConfigurer {
    private static final Logger logger = LoggerFactory.getLogger(MyRoutingDataSource.class);


    public static final String MASTER_DATASOURCE = "masterDataSource";
    public static final String SLAVE_DATASOURCE ="slaveDataSource";
    

    @Bean("masterDataSource")
    @ConfigurationProperties(prefix = "spring.datasource.master")
    public DruidDataSource getMasterDruidDataSource(){
        DruidDataSource build = DataSourceBuilder.create().type(DruidDataSource.class).build();
        logger.info("配置主数据库{}",build);
        return build;
    }

    @Bean("slaveDataSource")
    @ConfigurationProperties(prefix = "spring.datasource.slave")
    public DruidDataSource getSlaveDruidDataSource(){
        DruidDataSource build = DataSourceBuilder.create().type(DruidDataSource.class).build();
        logger.info("配置主数据库{}",build);
        return build;
    }
 


    /**
     * Primary 优先使用该Bean
     * DependsOn 先执行主从数据库的配置
     * Qualifier 指定使用哪个Bean
     *
     * @param masterDataSource
     * @param slaveDataSource
     * @return
     */

    @Bean
    @Primary
    @DependsOn(value = {MASTER_DATASOURCE, SLAVE_DATASOURCE})
    public DataSource routingDataSource(@Qualifier("masterDataSource") DruidDataSource  masterDataSource,
                                        @Qualifier("slaveDataSource") DruidDataSource  slaveDataSource){

        if(StringUtils.isEmpty(masterDataSource.getUrl())){
            logger.info("没有配置时默认使用master数据库");
            return masterDataSource;
        }
        Map<Object ,Object> map = new HashMap<>();
        map.put(MyDataSourceConfigurer.MASTER_DATASOURCE,masterDataSource);
        map.put(MyDataSourceConfigurer.SLAVE_DATASOURCE,slaveDataSource);
        MyRoutingDataSource routing = new MyRoutingDataSource();
        //设置动态数据源
        routing.setTargetDataSources(map);
        routing.setDefaultTargetDataSource(masterDataSource);
        logger.info("主从配置完成");
        return routing;
    }
}
import java.lang.annotation.*;

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE,ElementType.METHOD})
@Documented
public @interface DataSourceWith {
    String key() default "";
}
import com.gao.config.RoutingDataSourceHolder;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;

import java.lang.reflect.Method;

@Aspect
@Order(-1)
@Component
public class DataSourceWithAspect {

    @Pointcut(value = "@annotation(com.gao.aop.DataSourceWith) || @within(com.gao.aop.DataSourceWith)")
    public void myPointCut(){}


    @Before("myPointCut()")
    public void before(JoinPoint joinPoint){
        MethodSignature methodSignature = (MethodSignature)joinPoint.getSignature();
        Method method = methodSignature.getMethod();

        //获取注解对象
        DataSourceWith dataSource = method.getAnnotation(DataSourceWith.class);
        if(dataSource == null){
            //方法没有就获取类上的
            dataSource = method.getDeclaringClass().getAnnotation(DataSourceWith.class);
        }

        String key = dataSource.key();
        RoutingDataSourceHolder.setDataSource(key);

    }

    @After("myPointCut()")
    public void after(JoinPoint joinPoint){
        RoutingDataSourceHolder.clearDataSource();
    }
}

4、测试

@Controller
@RequestMapping("/hello")
public class MyController {

    @Resource
    private IUserService userService;

    @DataSourceWith(key = MyDataSourceConfigurer.SLAVE_DATASOURCE)
    @ResponseBody
    @GetMapping("/04")
    public List<User> getListUser(){
        return userService.getListUser();
    }

    @DataSourceWith(key = MyDataSourceConfigurer.MASTER_DATASOURCE)
    @ResponseBody
    @GetMapping("/03")
    public String addUser(@RequestParam("name") String name,@RequestParam("age") int age){
        userService.addUser(name,age);
        return "用户信息插入成功";
    }
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
分布式架构 漫谈分布式架构 初识分布式架构与意义 如何把应用从单机扩展到分布式 大型分布式架构演进过程 分布式架构设计 主流架构模型-SOA架构和微服务架构 领域驱动设计及业务驱动规划 分布式架构的基本理论CAP、BASE以及其应用 什么是分布式架构下的高可用设计 构架高性能的分布式架构 构建分布式架构最重要因素 CDN静态文件访问 分布式存储 分布式搜索引擎 应用发布与监控 应用容灾及机房规划 系统动态扩容 分布式架构策略-分而治之 从简到难,从网络通信探究分布式通信原理 基于消息方式的系统间通信 理解通信协议传输过程中的序列化和反序列化机制 基于框架的RPC通信技术 WebService/ApacheCXF RMI/Spring RMI Hession 传统RPC技术在大型分布式架构下面临的问题 分布式架构下的RPC解决方案 Zookeeper 分布式系统的基石 从0开始搭建3个节点额度zookeeper集群 深入分析Zookeeper在disconf配置中心的应用 基于Zookeeper Watcher 核心机制深入源码分析 Zookeeper集群升级、迁移 基于Zookeeper实现分布式服务器动态上下线感知 深入分析Zookeeper Zab协议及选举机制源码解读 Dubbo 使用Dubbo对单一应用服务化改造 Dubbo管理中心及及监控平台安装部署 Dubbo分布式服务模块划分(领域驱动) 基于Dubbo的分布式系统架构实战 Dubbo负载均衡策略分析 Dubbo服务调试之服务只订阅及服务只注册配置 Dubbo服务接口的设计原则(实战经验) Dubbo设计原理及源码分析 基于Dubbo构建大型分布式电商平台实战雏形 Dubbo容错机制及扩展性分析 分布式解决方案 分布式全局ID生成方案 session跨域共享及企业级单点登录解决方案实战 分布式事务解决方案实战 高并发下的服务降级、限流实战 基于分布式架构下分布式锁的解决方案实战 分布式架构实现分布式定时调度 分布式架构-中间件 分布式消息通信 消息中间件在分布式架构中的应用 ActiveMQ ActiveMQ高可用集群企业及部署方案 ActiveMQ P2P及PUB/SUB模式详解 ActiveMQ消息确认及重发策略 ActiveMQ基于Spring完成分布式消息队列实战 Kafka Kafka基于Zookeeper搭建高可用集群实战 kafka消息处理过程剖析 Java客户端实现Kafka生产者与消费者实例 kafka的副本机制及选举原理剖析 基于kafka实现应用日志实时上报统计分析 RabbitMQ 初步认识RabbitMQ及高可用集群部署 详解RabbitMQ消息分发机制及主题消息分发 RabbitMQ消息路由机制分析 RabbitMQ消息确认机制 Redis redis数据结构分析 Redis主从复制原理及无磁盘复制分析 Redis管道模式详解 Redis缓存与数据库一致性问题解决方案 基于redis实现分布式实战 图解Redis中的AOF和RDB持久化策略的原理 redis读写分离架构实践 redis哨兵架构及数据丢失问题分析 redis Cluster数据分布算法之Hash slot redis使用常见问题及性能优化思路 redis高可用及高并发实战 缓存击穿、缓存雪崩预防策略 Redis批量查询优化 Redis高性能集群之Twemproxy of Redis 数据存储 MongoDB NOSQL简介及MongoDB支持的数据类型分析 MongoDB可视化客户端及JavaApi实践 手写基于MongoDB的ORM框架 MongoDB企业级集解决方案 MongoDB聚合、索引及基本执行命令 MongoDB数据分片、转存及恢复策略 MyCat MySQL主从复制及读写分离实战 MySQL+keepalived实现双主高可用方案实践 MySQL高性能解决方案之分库分表 数据库中间件初始Mycat 基于Mycat实习MySQL数据库读写分离 基于Mycat实战之数据库切分策略剖析 Mycat全局表、Er表、分片预警分析 Nginx 基于OpenResty部署应用层Nginx以及Nginx+lua实战 Nginx反向代理服务器及负载均衡服务器配置实战 利用keepalived+Nginx实战Nginx高可用方案 基于Nginx实现访问控制、连接限制 Nginx动静分离实战 Nginx Location ReWrite 等语法配置及原理分析 Nginx提供https服务 基于Nginx+lua完成访问流量实时上报Kafka的实战 Netty 高性能NIO框架 IO 的基本概念、NIO、AIO、BIO深入分析 NIO的核心设计思想 Netty产生的背景及应用场景分析 基于Netty实现的高性能IM聊天 基于Netty实现Dubbo多协议通信支持 Netty无锁化串行设计及高并发处理机制 手写实现多协议RPC框架

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值