当前位置:Gxlcms > 数据库问题 > spring+mybatis+mysql5.7实现读写分离,主从复制

spring+mybatis+mysql5.7实现读写分离,主从复制

时间:2021-07-01 10:21:17 帮助过:6人阅读

bean id="writeDataSource" class="com.alibaba.druid.pool.DruidDataSource" init-method="init" destroy-method="close"> <description>写数据库连接</description> <property name="driverClassName" value="${db.driver}" /> <property name="url" value="${db.writer.url}" /> <property name="username" value="${db.writer.username}" /> <property name="password" value="${db.writer.password}" /> <!-- 配置初始化大小、最小、最大 --> <property name="initialSize" value="${db.writer.initialSize}" /> <property name="minIdle" value="${db.writer.maxIdle}" /> <property name="maxActive" value="${db.writer.maxActive}" /> <!-- 配置获取连接等待超时的时间 --> <property name="maxWait" value="${db.writer.maxWait}" /> <!-- 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒 --> <property name="timeBetweenEvictionRunsMillis" value="${db.timeBetweenEvictionRunsMillis}" /> <!-- 配置一个连接在池中最小生存的时间,单位是毫秒 --> <property name="minEvictableIdleTimeMillis" value="${db.minEvictableIdleTimeMillis}" /> <property name="validationQuery" value="SELECT 'x'" /> <property name="testWhileIdle" value="true" /> <property name="testOnBorrow" value="false" /> <property name="testOnReturn" value="false" /> <!-- 打开PSCache,并且指定每个连接上PSCache的大小 --> <property name="poolPreparedStatements" value="false" /> <property name="maxPoolPreparedStatementPerConnectionSize" value="20" /> <!-- 打开Druid的监控统计功能 --> <property name="filters" value="slf4j" /> <property name="proxyFilters"> <list> <ref bean="stat-filter" /> </list> </property> <property name="timeBetweenLogStatsMillis" value="300000" /> </bean>

读数据源配置

<bean id="readDataSource" class="com.alibaba.druid.pool.DruidDataSource" init-method="init" destroy-method="close">
        
        <description>只读数据库连接</description>
        <property name="driverClassName" value="${db.driver}" />
        <property name="url" value="${db.reader.url}" />
        <property name="username" value="${db.reader.username}" />
        <property name="password" value="${db.reader.password}" />

        <!-- 配置初始化大小、最小、最大 -->
        <property name="initialSize" value="${db.reader.initialSize}" />
        <property name="minIdle" value="${db.reader.maxIdle}" />
        <property name="maxActive" value="${db.reader.maxActive}" />

        <!-- 配置获取连接等待超时的时间 -->
        <property name="maxWait" value="${db.reader.maxWait}" />

        <!-- 配置间隔多久才进行一次检测,检测需要关闭的空闲连接,单位是毫秒 -->
        <property name="timeBetweenEvictionRunsMillis" value="${db.timeBetweenEvictionRunsMillis}" />

        <!-- 配置一个连接在池中最小生存的时间,单位是毫秒 -->
        <property name="minEvictableIdleTimeMillis" value="${db.minEvictableIdleTimeMillis}" />

        <property name="validationQuery" value="SELECT 'x'" />
        <property name="testWhileIdle" value="true" />
        <property name="testOnBorrow" value="false" />
        <property name="testOnReturn" value="false" />

        <!-- 打开PSCache,并且指定每个连接上PSCache的大小 -->
        <property name="poolPreparedStatements" value="false" />
        <property name="maxPoolPreparedStatementPerConnectionSize" value="20" />

        <!-- 打开Druid的监控统计功能 -->
        <property name="filters" value="slf4j" />
        <property name="proxyFilters">
            <list>
                <ref bean="stat-filter" />
            </list>
        </property>
        <property name="timeBetweenLogStatsMillis" value="300000" />
    </bean>

数据源配置

     <!--注意这里的配置-->
    <bean id="dataSource" class="com.cht.integration.aspect.ChooseDataSource" lazy-init="true">
        <description>数据源</description>
        <property name="targetDataSources">
            <map key-type="java.lang.String" value-type="javax.sql.DataSource">
                <!-- write -->
                <entry key="write" value-ref="writeDataSource" />
                <!-- read -->
                <entry key="read" value-ref="readDataSource" />
            </map>
        </property>
        <property name="defaultTargetDataSource" ref="writeDataSource" />
        <property name="methodType">
            <map key-type="java.lang.String">
                <!-- read -->
                <entry key="read" value=",get,select,count,list,query," />
                <!-- write -->
                <entry key="write" value=",add,insert,create,update,delete,remove," />
            </map>
        </property>
    </bean>

数据源切面配置

    <!-- 注意这个切面 -->
    <bean id="DataSourceAspect" class="com.cht.integration.aspect.DataSourceAspect" />

获取数据源代码

/**
 * 获取数据源
 *
 * @author
 * @version
 */
public class ChooseDataSource extends AbstractRoutingDataSource {
    public static Map<String, List<String>> METHODTYPE = new HashMap<String, List<String>>();

    // 获取数据源名称
    protected Object determineCurrentLookupKey() {
        return HandleDataSource.getDataSource();
    }

    // 设置方法名前缀对应的数据源
    public void setMethodType(Map<String, String> map) {
        for (String key : map.keySet()) {
            List<String> v = new ArrayList<String>();
            String[] types = map.get(key).split(",");
            for (String type : types) {
                if (StringUtils.isNotBlank(type)) {
                    v.add(type);
                }
            }
            METHODTYPE.put(key, v);
        }
    }
}

AOP切换数据源配置


/**
 * 切换数据源(不同方法调用不同数据源)
 *
 * @author
 * @version
 */
@Aspect
@EnableAspectJAutoProxy(proxyTargetClass = true)
public class DataSourceAspect {
    private final Logger logger = LoggerFactory.getLogger(DataSourceAspect.class);
    //配置切入点
   // @Pointcut("execution(* com.zhx.service..*.*(..))")
    public void aspect() {
    }

    /**
     * 配置前置通知,使用在方法aspect()上注册的切入点
     */
   // @Before("com.cht.integration.aspect()")
    public void before(JoinPoint point) {
            String className = point.getTarget().getClass().getName();
        String method = point.getSignature().getName();
        logger.info(className + "." + method + "(" + StringUtils.join(point.getArgs(), ",") + ")");
        try {
            L: for (String key : ChooseDataSource.METHODTYPE.keySet()) {
                for (String type : ChooseDataSource.METHODTYPE.get(key)) {
                    if (method.startsWith(type)) {
                        logger.info(key);
                        HandleDataSource.putDataSource(key);
                        break L;
                    }
                }
            }
        } catch (Exception e) {
            logger.error(e.getMessage(), e);
            HandleDataSource.putDataSource("write");
        }
    }

   // @After("com.cht.integration.aspect()")
    public void after(JoinPoint point) {
        HandleDataSource.clear();
    }
}

数据库配置

主数据库配置

注意:务必保持主从数据库是同一个版本的数据库,否则容易出现兼容问题。
从数据库,一般用于存储,具体配置步骤如下:
1.找到my.ini目录所在位置,默认如下:
技术图片

2.修改对应的配置。

//server-id 必须保证唯一,不能重复。
server-id=1
//指定二进制日志文件的存储路径和名称
log-bin=mysql-bin

重启mysql主服务

查看主服务器master信息 show master status
技术图片

从服务器配置

1.找到my.ini目录所在位置,默认如下:
技术图片
2.修改从服务器配置,如下:

server-id=2
log-bin=mysql-bin
relay_log=mysql-relay-bin
log_slave_updates=1
read_only=1

3.重启从服务器服务
4.如果之前设置过从服务器配置,需要先 reset slave; ,否则跳过这一步
5.指定从服务器的主服务器

change master to master_host='192.168.159.128', --指定主服务器的IP地址
master_port=3306, --指定主服务器的端口
master_use='root', --指定主服务器的操作用户
master_password='root', --指定主服务器的密码
master_log_file='mysql-bin.000003', --指定要同步主服务器的日志文件
master_log_pos='154'; --指定要同步的节点,和show master status 中的 position值保持一致

6.启动从服务器配置 start slave;

7.show slave statusG; 查看从服务状态
技术图片
图上标志的两个都为yes时,则标示配置成功

测试是否同步

修改主服务器的数据,会发现数据同步到从服务器中,我这里是测试成功的,具体的就不截图了,如果没同步成功,那么请检查是否主从服务器版本是否为同一个版本,然后在检查配置是否写错,主从同步延时那个就参考其他资料了,这就先介绍到这里。

参考博客:https://blog.csdn.net/juded/article/details/54600294

spring+mybatis+mysql5.7实现读写分离,主从复制

标签:能力   原理   private   creat   statement   service   不同方法   读写分离   列表   

人气教程排行