手搓一个不用中间件的分表策略

++场景:针对一些特别的项目,不用中间件,以月为维度进行分表,代码详细设计方案++

1. 定义分片策略

首先,定义一个分片策略类,用于决定数据存储在哪个分表中

java 复制代码
import java.time.LocalDate;
import java.time.format.DateTimeFormatter;

public class ShardingStrategy {
    private static final DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy_MM");

    public String getShardId(LocalDate date) {
        return date.format(formatter);
    }
}

2. 数据库连接管理

创建一个数据库连接管理类,用于管理数据库连接

java 复制代码
import java.sql.Connection;
import java.sql.DriverManager;

public class DatabaseManager {
    private static Connection connection;

    static {
        try {
            Class.forName("com.mysql.cj.jdbc.Driver");
            connection = DriverManager.getConnection("jdbc:mysql://localhost:3306/mydatabase", "user", "password");
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    public static Connection getConnection() {
        return connection;
    }
}

3. 动态创建表

创建一个方法,用于根据当前月份动态创建新的表

java 复制代码
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.SQLException;
import java.time.LocalDate;

public class TableCreator {
    private ShardingStrategy shardingStrategy = new ShardingStrategy();

    public void createTableForMonth(LocalDate date) {
        String shardId = shardingStrategy.getShardId(date);
        String tableName = "orders_" + shardId;
        Connection connection = DatabaseManager.getConnection();

        String createTableSQL = "CREATE TABLE IF NOT EXISTS " + tableName + " ("
                + "id BIGINT PRIMARY KEY,"
                + "order_date DATE,"
                + "customer_name VARCHAR(255)"
                + ")";

        try (PreparedStatement statement = connection.prepareStatement(createTableSQL)) {
            statement.execute();
            System.out.println("Table " + tableName + " created successfully.");
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }
}

4. 订单操作类

创建一个订单操作类,用于执行具体的数据库操作。这里根据分片策略动态生成表名

java 复制代码
import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.time.LocalDate;

public class OrderDAO {
    private ShardingStrategy shardingStrategy = new ShardingStrategy();
    private TableCreator tableCreator = new TableCreator();

    public void createOrder(long orderId, LocalDate orderDate, String customerName) {
        // 确保表存在
        tableCreator.createTableForMonth(orderDate);

        String shardId = shardingStrategy.getShardId(orderDate);
        String tableName = "orders_" + shardId;
        Connection connection = DatabaseManager.getConnection();

        try (PreparedStatement statement = connection.prepareStatement(
                "INSERT INTO " + tableName + " (id, order_date, customer_name) VALUES (?, ?, ?)")) {
            statement.setLong(1, orderId);
            statement.setDate(2, java.sql.Date.valueOf(orderDate));
            statement.setString(3, customerName);
            statement.executeUpdate();
        } catch (SQLException e) {
            e.printStackTrace();
        }
    }

    public String getCustomerNameByOrderId(long orderId, LocalDate orderDate) {
        String shardId = shardingStrategy.getShardId(orderDate);
        String tableName = "orders_" + shardId;
        Connection connection = DatabaseManager.getConnection();

        try (PreparedStatement statement = connection.prepareStatement(
                "SELECT customer_name FROM " + tableName + " WHERE id = ?")) {
            statement.setLong(1, orderId);
            ResultSet resultSet = statement.executeQuery();
            if (resultSet.next()) {
                return resultSet.getString("customer_name");
            }
        } catch (SQLException e) {
            e.printStackTrace();
        }
        return null;
    }
}

5. 使用示例

最后,展示如何使用上述类来创建订单和查询订单信息

java 复制代码
public class Main {
    public static void main(String[] args) {
        OrderDAO orderDAO = new OrderDAO();

        // 创建订单
        orderDAO.createOrder(1, LocalDate.of(2023, 1, 15), "Alice");
        orderDAO.createOrder(2, LocalDate.of(2023, 2, 20), "Bob");
        orderDAO.createOrder(3, LocalDate.of(2023, 3, 10), "Charlie");

        // 查询订单
        System.out.println("Order ID 1: " + orderDAO.getCustomerNameByOrderId(1, LocalDate.of(2023, 1, 15)));
        System.out.println("Order ID 2: " + orderDAO.getCustomerNameByOrderId(2, LocalDate.of(2023, 2, 20)));
        System.out.println("Order ID 3: " + orderDAO.getCustomerNameByOrderId(3, LocalDate.of(2023, 3, 10)));
    }
}

总结

  • 分片策略:根据订单日期决定数据存储在哪个分表中。
  • 数据库连接管理:管理数据库连接,所有分表都在同一个数据库中。
  • 动态创建表:根据当前月份动态创建新的表。
  • 订单操作:执行具体的数据库操作,如插入和查询。通过动态生成表名来实现分表。
  • 使用示例:展示如何使用这些类来实现分表功能。

这个方案适用于在同一数据库中按月份进行分表的需求,可以根据实际业务场景进一步扩展和优化。

相关推荐
苏渡苇3 小时前
优雅应对异常,从“try-catch堆砌”到“设计驱动”
java·后端·设计模式·学习方法·责任链模式
团子的二进制世界3 小时前
G1垃圾收集器是如何工作的?
java·jvm·算法
long3163 小时前
Aho-Corasick 模式搜索算法
java·数据结构·spring boot·后端·算法·排序算法
rannn_1114 小时前
【苍穹外卖|Day4】套餐页面开发(新增套餐、分页查询、删除套餐、修改套餐、起售停售)
java·spring boot·后端·学习
灵感菇_4 小时前
Java HashMap全面解析
java·开发语言
qq_12498707534 小时前
基于JavaWeb的大学生房屋租赁系统(源码+论文+部署+安装)
java·数据库·人工智能·spring boot·计算机视觉·毕业设计·计算机毕业设计
短剑重铸之日4 小时前
《设计模式》第十一篇:总结
java·后端·设计模式·总结
若鱼19194 小时前
SpringBoot4.0新特性-Observability让生产环境更易于观测
java·spring
觉醒大王4 小时前
强女思维:着急,是贪欲外显的相。
java·论文阅读·笔记·深度学习·学习·自然语言处理·学习方法
努力学编程呀(๑•ี_เ•ี๑)4 小时前
【在 IntelliJ IDEA 中切换项目 JDK 版本】
java·开发语言·intellij-idea