java 数据库读写分离

                                      java  数据库读写分离

最近有个新任务,研究数据库读写分离,网上找了很多,有不成熟的也有成熟的,不过总是有这个那个的问题,这个在汇总一遍。 数据库读写分离的优势显而易见,可以大大减轻服务器压力,写用master数据库,读用slave数据库,在查找资料的过程中发现了几种方式。

  1. 列表内容

    1).利用MySQL自带的功能实现读写分离
    查找资料过程中发现有大神提到可以直接使用MySQL自带的功能来解决读写分离,后来尝试了一下,确实能行。

    数据源:

jdbc.driverClassName=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql:replication://192.168.20.9:3306,192.168.20.8:3306/gm_dev?zeroDateTimeBehavior=convertToNull&useUnicode=true&characterEncoding=UTF-8
jdbc.search.url=jdbc:mysql:replication://192.168.20.9:3306,192.168.20.8:3306/gsearch08?zeroDateTimeBehavior=convertToNull&useUnicode=true&characterEncoding=UTF-8
jdbc.old.url=jdbc:mysql:replication://192.168.20.9:3306,192.168.20.8:3306/gm_old?zeroDateTimeBehavior=convertToNull&useUnicode=true&characterEncoding=UTF-8
jdbc.username=xxxxxx
jdbc.password=xxxxxxxxxx
jdbc.search.username=xxxxxxxxxxx
jdbc.search.password=xxxxxxxxx
jdbc.old.username=xxxxxxx
jdbc.old.password=xxxxxxxx
site.conf.path=xxxxxxx

AOP事务传播,read-only="true" 是slave数据库,其余的是master数据库

如果一个service方法里,既有读的,又有写的,MYSQL还是可以读的用slave,写的用master数据库
优点:方便,简单.
缺点:不够灵活。
我这儿还有个问题,如果将切面定在service层,读写可以分离,不过放在dao层,读写不能分离,不知道是什么原因。

2)spring AOP 解决读写分离

具体原理可以看下http://www.cnblogs.com/surge/p/3582248.html

数据源

jdbc.driverClassName=com.mysql.jdbc.Driver
master.jdbc.url=jdbc:mysql://192.168.20.9:3306/gm_dev?zeroDateTimeBehavior=convertToNull&useUnicode=true&characterEncoding=UTF-8
master.jdbc.search.url=jdbc:mysql://192.168.20.9:3306/gsearch08?zeroDateTimeBehavior=convertToNull&useUnicode=true&characterEncoding=UTF-8
master.jdbc.old.url=jdbc:mysql://192.168.20.9:3306/gm_old?zeroDateTimeBehavior=convertToNull&useUnicode=true&characterEncoding=UTF-8
slave.jdbc.url=jdbc:mysql://192.168.20.8:3306/gm_dev?zeroDateTimeBehavior=convertToNull&useUnicode=true&characterEncoding=UTF-8
slave.jdbc.search.url=jdbc:mysql://192.168.20.8:3306/gsearch08?zeroDateTimeBehavior=convertToNull&useUnicode=true&characterEncoding=UTF-8
slave.jdbc.old.url=jdbc:mysql://192.168.20.8:3306/gm_old?zeroDateTimeBehavior=convertToNull&useUnicode=true&characterEncoding=UTF-8

java代码:
package com.gotobus.common;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface DataSource {
    String value();
}


-----------------------------------------------------------------------------------------------------------



package com.gotobus.common;

import org.aspectj.lang.JoinPoint;

import java.lang.reflect.Method;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.reflect.MethodSignature;


public class DataSourceAspect {
    public static final ThreadLocal<String> holder = new ThreadLocal<String>();

    public void before(JoinPoint point) {
        Object target = point.getTarget();
        String method = point.getSignature().getName();
        Class<?>[] parameterTypes = ((MethodSignature) point.getSignature())
                .getMethod().getParameterTypes();
        try {
            Method m = target.getClass().getMethod(method, parameterTypes);
            if (m != null && m.isAnnotationPresent(DataSource.class)) {
                DataSource data = m
                        .getAnnotation(DataSource.class);
                putDataSource(data.value());
                System.out.println(data.value());
            }

        } catch (Exception e) {
            // TODO: handle exception
        }
    }

    public static void putDataSource(String name){
        holder.set(name);
    }

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

 1. 列表内容

-------------------------------------------------------------------------------------

package com.gotobus.common;

import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource;

public class DynamicDataSource extends AbstractRoutingDataSource{

@Override
protected Object determineCurrentLookupKey() {
    Object dataSource=DataSourceAspect.getDataSource();
    return dataSource;
}

}


  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java连接主备数据库进行分离可以使用JDBC的方式进行连接,具体步骤如下: 1. 引入JDBC驱动程序,例如MySQL Connector/J; 2. 创建主库和备库的连接URL,格式为jdbc:mysql://host:port/database,其中host和port分别为主库和备库的IP地址和端口号,database为数据库名称; 3. 创建主库和备库的连接用户名和密码; 4. 根据需要发送请求或请求,使用Connection、PreparedStatement、ResultSet等JDBC对象进行操作。 下面是一个使用Java连接主备数据库进行分离的示例代码: ```java import java.sql.Connection; import java.sql.DriverManager; import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.SQLException; public class MySQLReadWriteSplittingExample { public static void main(String[] args) { String masterUrl = "jdbc:mysql://master-server:3306/mydatabase"; String slaveUrl = "jdbc:mysql://slave-server:3306/mydatabase"; String user = "myuser"; String password = "mypassword"; Connection masterConnection = null; Connection slaveConnection = null; PreparedStatement statement = null; ResultSet resultSet = null; try { // 连接主库 masterConnection = DriverManager.getConnection(masterUrl, user, password); // 发送请求 statement = masterConnection.prepareStatement("SELECT * FROM mytable"); resultSet = statement.executeQuery(); // 处理查询结果 while (resultSet.next()) { int id = resultSet.getInt("id"); String name = resultSet.getString("name"); System.out.println("ID: " + id + ", Name: " + name); } // 连接备库 slaveConnection = DriverManager.getConnection(slaveUrl, user, password); // 发送请求 statement = slaveConnection.prepareStatement("INSERT INTO mytable (id, name) VALUES (?, ?)"); statement.setInt(1, 1); statement.setString(2, "John Doe"); statement.executeUpdate(); } catch (SQLException e) { e.printStackTrace(); } finally { // 释放资源 try { if (resultSet != null) { resultSet.close(); } if (statement != null) { statement.close(); } if (masterConnection != null) { masterConnection.close(); } if (slaveConnection != null) { slaveConnection.close(); } } catch (SQLException e) { e.printStackTrace(); } } } } ``` 在上面的示例代码中,我们创建了主库连接和备库连接的URL、用户名和密码,然后分别使用这两个连接对象进行了操作和操作。由于使用的是两个独立的连接对象,因此需要分别释放资源。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值