1

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

      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

最好的问候 - 乌尔里希

4

4 回答 4

2

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

于 2021-02-10T21:43:41.407 回答
1

DATE_FORMAT(expr, format)不能在虚拟列中使用,因为它取决于连接的语言环境(MariaDB 问题MDEV-11553)。

为添加语言环境的date_format创建了一个 3 参数形式。

DATE_FORMAT(visit_date, '%Y-%m-%d', 'en_US')可以在 MariaDB-10.3+ 稳定版本的虚拟列表达式中使用。

绝对建议使用DATE或更改查询以不使用列表达式周围的函数。

于 2021-02-11T05:14:29.093 回答
1

这对你有用吗?

SELECT t.t_date td, v.visit_date vd
  FROM temp_dates t
  LEFT JOIN visits v ON DATE(v.visit_date) = DATE(t.t_date)
 ORDER BY t.t_date

如果是这样,您的问题有一个可行的解决方案:

  1. 使用对象上的确定性函数添加一DATE列。像这样。DATE()visit_date

    ALTER TABLE visits ADD COLUMN dateval DATE AS (DATE(visit_date)) VIRTUAL; 
    CREATE INDEX idx_visit_date on visits(dateval);
    
  2. 然后在另一个表中创建一个虚拟列(将格式精美的日期塞入您的 VARCHAR() 列中的那个。

    ALTER TABLE temp_dates ADD COLUMN dateval DATE AS (DATE(t_date)) VIRTUAL;
    CREATE INDEX idx_temp_dates_date on temp_dates (dateval);
    

这是有效的,因为DATE()它是确定性的,不像DATE_FORMAT().

那么你的查询应该是。

SELECT t.t_date td, v.visit_date vd
  FROM temp_dates t
  LEFT JOIN visits v ON v.dateval = t.dateval
 ORDER BY t.t_date

此解决方案为您提供(虚拟)DATE列的索引。这很好,因为此类列上的索引匹配是有效的。

但是,您最好的解决方案是将temp_date.t_datefrom的数据类型更改VARCHAR()DATE

于 2021-02-10T22:16:17.793 回答
0

函数不是“可存储的”。

考虑:

ON  v.visit_date >= t.t_date
AND v.visit_date  < t.t_date + INTERVAL 1 DAY
于 2021-02-12T02:29:55.443 回答