小编典典

最近7天的MySQL计数数据

sql

我有以下架构。

表票

+------------------+--------------+------+-----+---------------------+----------------+
| Field            | Type         | Null | Key | Default             | Extra          |
+------------------+--------------+------+-----+---------------------+----------------+
| id               | int(10)      | NO   | PRI | NULL                | auto_increment |
| aid              | varchar(10)  | NO   |     |                     |                |
| ip               | varchar(100) | NO   |     |                     |                |
| host             | varchar(200) | NO   |     |                     |                |
| timestamp        | varchar(20)  | NO   |     | 0000-00-00 00:00:00 |                |
| user             | tinytext     | NO   |     | NULL                |                |
| userid           | int(10)      | NO   |     | 0                   |                |
+------------------+--------------+------+-----+---------------------+----------------+

在这里,我想获取最近7天中每天的每种援助的计数,在没有任何援助票的日期为“ 0”。时间戳是unix时间戳。

任何帮助都将受到高度赞赏。


阅读 304

收藏
2021-03-23

共1个答案

小编典典

MySQL没有递归功能,因此您可以使用NUMBERS表技巧-

  1. 创建仅包含递增数字的表-使用auto_increment可以轻松做到:

    DROP TABLE IF EXISTS `example`.`numbers`;
    

    CREATE TABLE example.numbers (
    id int(10) unsigned NOT NULL auto_increment,
    PRIMARY KEY (id)
    ) ENGINE=InnoDB DEFAULT CHARSET=latin1;

  2. 使用以下命令填充表:

    INSERT INTO NUMBERS
    

    (id)
    VALUES
    (NULL)

…根据需要提供尽可能多的值。

  1. 使用DATE_ADD构造日期列表,并根据NUMBERS.id值增加日期。用您各自的开始和结束日期替换“ 2010-01-01”和“ 2010-01-02”(但使用相同的格式,YYYY-MM-DD HH:MM:SS)。在此示例中,我从CURRENT_DATE中减去了NUMBERS.id值,以获取上周的顺序日期值列表-

    SELECT x.dt
    

    FROM (SELECT DATE_SUB(CURRENT_DATE, INTERVAL (n.id - 1) DAY) AS dt
    FROM numbers n
    WHERE n.id <= 7 ) x

  2. 根据日期时间部分将其左联接到数据表中。

       SELECT x.dt,
           COUNT(v.aid) AS num
     FROM (SELECT DATE_SUB(CURRENT_DATE, INTERVAL (n.id - 1) DAY) AS dt
             FROM numbers n
            WHERE n.id <= 7 ) x
    

    LEFT JOIN VOTES v ON DATE(FROM_UNIXTIME(v.timestamp)) = DATE(x.dt)
    GROUP BY x.dt
    ORDER BY x.dt

为什么用数字而不是日期?

简单-可以根据数字生成日期,就像我提供的示例一样。这也意味着只使用一个表,而不是每个数据类型一个表。

之前:

  SELECT DATE(FROM_UNIXTIME(v.timestamp)) AS dt,
         COUNT(v.aid)
    FROM VOTES v
   WHERE DATE(FROM_UNIXTIME(v.timestamp)) BETWEEN DATE_SUB(CURRENT_DATE, INTERVAL 7 DAY)
                                              AND CURRENT_DATE
GROUP BY DATE(FROM_UNIXTIME(v.timestamp))
2021-03-23