postgreSQL创建表分区

postgreSQL创建表分区

1.范围分区介绍

复制代码
范围分区是根据某个列的范围来进行分区。常用于日期、时间或其他连续值类型的列。例如,按日期字段对表进行分区。

2.创建表

复制代码
创建分区时使用的复合主键,所以在代码中如果集成了mybatisplus,不要使用selectById等类似方法,而是使用QueryWrapper查询。
sql 复制代码
CREATE TABLE "public"."car" (
  "id" varchar(64) COLLATE "pg_catalog"."default" NOT NULL,
  "plate_no" varchar(255) COLLATE "pg_catalog"."default",
  "create_time" timestamp(6),
  "car_id" varchar(64) COLLATE "pg_catalog"."default",
  "height" varchar(64) COLLATE "pg_catalog"."default",
  CONSTRAINT "car_partition_pkey" PRIMARY KEY ("id","create_time")
) PARTITION BY RANGE (
  "create_time" "pg_catalog"."timestamp_ops"
)
;

3.每月创建分区函数

sql 复制代码
CREATE OR REPLACE FUNCTION "public"."monthly_partition"()
  RETURNS "pg_catalog"."void" AS $BODY$
DECLARE
    start_date TIMESTAMP;
    end_date TIMESTAMP;
    partition_name TEXT;
		partition_date TEXT;
BEGIN
    -- 获取当前日期的下个月的第一天
    start_date := date_trunc('month', CURRENT_DATE) + INTERVAL '1 month';
    -- 获取下个月的最后一天
    end_date := start_date + INTERVAL '1 month';

    -- 动态生成car分区表的名称
    partition_name := 'car_' || TO_CHAR(start_date, 'YYYY_MM');
    
    -- 创建car分区表
    EXECUTE format(
        'CREATE TABLE IF NOT EXISTS %I PARTITION OF car FOR VALUES FROM (%L) TO (%L)',
        partition_name, start_date, end_date
    );
                
                    -- 动态生成car分区表的名称
    partition_name := 'car_' || TO_CHAR(start_date, 'YYYY_MM');
    
    -- 创建car分区表
    EXECUTE format(
        'CREATE TABLE IF NOT EXISTS %I PARTITION OF car FOR VALUES FROM (%L) TO (%L)',
        partition_name, start_date, end_date
    );
		
		-- 为分区动态创建索引
        EXECUTE format(
            'CREATE INDEX IF NOT EXISTS plate_no_car_id_idx_%s ON %I (
                plate_no,
                car_id
            )',
            partition_date, partition_name
        );
    

END;
$BODY$
  LANGUAGE plpgsql VOLATILE
  COST 100;

4.调用函数

在数据库中执行一次

sql 复制代码
SELECT monthly_partition();

注:如果每月创建一个分区,可以通过定时任务调用数据库函数。(通过定时任务调用,每月执行一次)

在mapper中定义方法

java 复制代码
String createMonthlyPartition();

然后在xml中编写sql

xml 复制代码
    <select id="createMonthlyPartition" statementType="CALLABLE" resultType="java.lang.String" >
        {call monthly_partition()}
    </select>

5.历史数据按月创建分区

如果旧表没有创建分区,重新建表分区后,必须要根据时间字段创建分区后才能导入数据。创建分区可以执行下面脚本。(只需执行一次)

sql 复制代码
DO $$
DECLARE
    start_date DATE := '2024-01-01';  -- 起始日期,建议设置为月份的第一天
    current_date DATE := CURRENT_DATE; -- 当前日期
    partition_name TEXT;
    start_time TIMESTAMP;
    end_time TIMESTAMP;
    d DATE;  -- 声明变量 d 为 DATE 类型
BEGIN
    -- 使用 generate_series 生成从 start_date 到 current_date 的月份范围
    FOR d IN 
        SELECT date_trunc('month', generate_series(start_date, current_date, '1 month'::INTERVAL))
    LOOP
        -- 为每个月生成对应的时间范围
        start_time := d;
        end_time := d + INTERVAL '1 month';

        -- 动态生成分区名称
        partition_name := 'car_' || TO_CHAR(d, 'YYYY_MM');

        -- 创建分区
        EXECUTE format(
            'CREATE TABLE IF NOT EXISTS %I PARTITION OF car 
             FOR VALUES FROM (%L) TO (%L)',
            partition_name, start_time, end_time
        );
    END LOOP;
END $$;

6.查看、删除分区索引

查看分区索引

sql 复制代码
SELECT * FROM pg_indexes WHERE tablename = 'car_2025_01';

删除分区索引。plate_no_car_id_idx_为索引名

sql 复制代码
DROP INDEX IF EXISTS plate_no_car_id_idx_;
相关推荐
柯南二号1 小时前
【Java后端】Spring Boot 集成 MyBatis-Plus 全攻略
java·spring boot·mybatis
星霜笔记3 小时前
Docker 部署 MariaDB+phpMyAdmin+Nextcloud 完整教程
运维·数据库·docker·容器·mariadb
桦说编程8 小时前
Java 中如何创建不可变类型
java·后端·函数式编程
lifallen8 小时前
Java Stream sort算子实现:SortedOps
java·开发语言
IT毕设实战小研8 小时前
基于Spring Boot 4s店车辆管理系统 租车管理系统 停车位管理系统 智慧车辆管理系统
java·开发语言·spring boot·后端·spring·毕业设计·课程设计
wyiyiyi8 小时前
【Web后端】Django、flask及其场景——以构建系统原型为例
前端·数据库·后端·python·django·flask
没有bug.的程序员9 小时前
JVM 总览与运行原理:深入Java虚拟机的核心引擎
java·jvm·python·虚拟机
天宇_任9 小时前
Mysql数据库迁移到GaussDB注意事项
数据库·mysql·gaussdb
甄超锋9 小时前
Java ArrayList的介绍及用法
java·windows·spring boot·python·spring·spring cloud·tomcat
阿华的代码王国10 小时前
【Android】RecyclerView复用CheckBox的异常状态
android·xml·java·前端·后端