首页 > 解决方案 > MySQL:优化格式化日期的左连接

问题描述

我正在尝试优化此查询的速度:

      SELECT t.t_date td, v.visit_date vd
      FROM temp_dates t
      LEFT JOIN visits v ON DATE_FORMAT(v.visit_date, '%Y-%m-%d') = t.t_date
      ORDER BY t.t_date

v.visit_date 是 DATETIME 类型,t.t_date 是格式为 '%Y-%m-%d' 的字符串。简单地在 v.visitdate 上创建索引并没有提高速度。因此,我打算尝试@oysteing 在这里给出的解决方案: How to optimize mysql group by with DATE_FORMAT 我成功地通过此 SQL 创建了一个虚拟列 ALTER TABLE visits ADD COLUMN datestr varchar(10) AS (DATE_FORMAT(visit_date, '%Y-%m-%d')) VIRTUAL; 但是当我尝试在此列上创建索引时, CREATE INDEX idx_visit_date on visits(datestr)我收到此错误:

#1901 - 函数或表达式 'date_format()' 不能在 GENERATED ALWAYS AS 子句中使用datestr

我究竟做错了什么?我的数据库是 Maria DB 10.4.8

最好的问候 - 乌尔里希

标签: mysqldatetimemariadbquery-optimizationvirtual-column

解决方案


date_format()也不能用于持久生成的列。而且在索引中它不能只是虚拟的,它必须被持久化。

我在手册中找不到明确的声明,但我相信这是因为输出date_format()可以取决于语言环境,因此不是严格确定的。

date_format()您可以使用确定性函数(例如concat()year()、和)来构建字符串month(),而不是。day()lpad()

...
datestr varchar(10) AS (concat(year(visit_date),
                               '-',
                               lpad(month(visit_date), 2, '0'),
                               '-',
                               lpad(day(visit_date), 2, '0')))
...

但正如我在评论中已经提到的那样,你正在修复错误的结局。日期/时间不应存储为字符串。因此,您应该提升temp_dates.t_date为 adate并使用date()来提取生成的索引列中的date部分visit_date

...
visit_date_date date AS (date(visit_date))
...

而且您可能还想尝试索引temp_dates.t_date


推荐阅读