LEFT JOIN之后的MySQL查询计算问题

时间:2018-10-28 07:21:48

标签: mysql sql join sum left-join

我有两个表 user_point_logs user_point_used_logs 从那里我想得到这样的输出

╔══════╦═════╦══════════════════════╦═══════════════════════╗══════════════╗
║ year ║month║ user_next_expire_date║ user_point_next_expire║minus         ║
╠══════╬═════╬══════════════════════╬═══════════════════════╣══════════════╣
║2018  ║ 10  ║ 2018年10月27日        ║ 200                   ║110          ║
║      ║     ║                      ║                       ║              ║ 
╚══════╩═════╩══════════════════════╩═══════════════════════╝══════════════╝
  

这是关系user_point_logs.return_id = user_point_used_logs.id   和user_point_used_logs.user_point_log_used_id = user_point_logs.id

user_point_logs

╔════╦════════╦═════╦═══════════╦══════════╗
║ id ║user_id ║point║expire_date║return_id ║
╠════╬════════╬═════╬═══════════╬══════════╣
║  1 ║ 16087  ║ 100 ║1540649153 ║   null   ║
║  2 ║ 16087  ║ 100 ║ null      ║   1      ║
║  3 ║ 16087  ║ 10  ║ null      ║   2      ║
║  4 ║ 16087  ║ 100 ║ 1540649153║   null   ║ 
╚════╩════════╩═════╩═══════════╩══════════╝

user_point_used_logs

|-------|-----------------------|-----------------------|------|
| Id    | user_point_log_id     | user_point_log_used_id| point|
|-------|-----------------------|-----------------------|------|
| 1     |         2             |         2             | 100  |
|-------|-----------------------|-----------------------|------|
| 2     |         3             |         3             | 10   |
|-------|-----------------------|-----------------------|------|

我只是尝试放置一些示例数据。 这是我的查询。

SELECT 
  YEAR(FROM_UNIXTIME(user_point_logs.expire_date, '%Y-%m-%d')) AS year, 
  MONTH(FROM_UNIXTIME(user_point_logs.expire_date, '%Y-%m-%d')) AS month, 
  FROM_UNIXTIME(
    SUBSTRING_INDEX(
      GROUP_CONCAT(
        DISTINCT user_point_logs.expire_date ORDER BY user_point_logs.expire_date ASC
        SEPARATOR ','
      ), 
      ',', 
      1
    ), 
    '%Y年%m月%d日'
  ) AS user_next_expire_date, 
  SUM(user_point_logs.point) AS user_point_next_expire, 
  IF(
    SUM(`user_point_used_logs`.`point`) IS NULL, 
    0, 
    SUM(`user_point_used_logs`.`point`) 
  ) AS `minus` 
FROM `user_point_logs` 
LEFT JOIN (
  SELECT *
  FROM user_point_used_logs
  WHERE user_point_log_id NOT IN (
    SELECT DISTINCT return_id
    FROM user_point_logs
    WHERE return_id IS NOT NULL
  )
) AS user_point_used_logs 
ON (`user_point_logs`.`id` = `user_point_used_logs`.`user_point_log_used_id`) 
WHERE `user_point_logs`.`user_id` = '16087' 
AND `user_point_logs`.`point` > 0
AND `user_point_logs`.`expire_date` > 1540464465 
AND (
  `user_point_logs`.`return_id` = 0 OR 
  `user_point_logs`.`return_id` IS NULL
)
GROUP BY `year`, `month`

问题在于,有时剩余点和减去点的计算没有得到实际结果。在我的小提琴中,您将看到user_point_next_expire 200但减0。 但是应该减为100。

请告诉我我的查询是否有问题。

这是我的示例数据集...

CREATE TABLE user_point_logs (
  id          int NOT NULL,
  user_id     int NOT NULL,
  point       int(11) NOT NULL,
  expire_date int DEFAULT NULL,
  return_id   int DEFAULT NULL
);

CREATE TABLE user_point_used_logs (
  id                      int NOT NULL,
  user_point_log_id       int NOT NULL,
  user_point_log_used_id  int DEFAULT '0' ,
  point                   int NOT NULL 
);

INSERT INTO user_point_used_logs 
  (id,  user_point_log_id,  user_point_log_used_id, point) 
VALUES 
  (1,   2,                  2,                      100),
  (2,   3,                  3,                      10);

INSERT INTO user_point_logs 
  (id,  user_id,  point,  expire_date,  return_id) 
VALUES
  (1,   16087,    100,    1540649153,   NULL),
  (2,   16087,    100,    NULL      ,   1   ),
  (3,   16087,     10,    NULL      ,   2   ),
  (4,   16083,    100,    1540649153,   NULL),
  (5,   16087,    100,    1540649153,   NULL);

..和fiddle相同

1 个答案:

答案 0 :(得分:1)

这个问题的问题是我看不到以下问题无法解决的问题的一部分:

SELECT MAX(FROM_UNIXTIME(expire_date)) dt
     , SUM(CASE WHEN expire_date < UTC_TIMESTAMP() THEN point END) a
     , SUM(CASE WHEN expire_date IS NULL THEN point END) b
  FROM user_point_logs
 WHERE user_id = 16087
 GROUP 
    BY user_id;