繁体   English   中英

过滤所有行的特定条件,并为具有相同值的每一列仅返回一行

[英]Filtering down specific criteria for all rows and return only one row for each column with the same value

我有house_leaseshouse_lease_terms (参见下面的表格模式)。 一个house_lease可以有多个house_lease_term,但一次只能有一个“活动”期限。

表定义:

CREATE TABLE `house_leases` (
  `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
  `house_id` int(10) unsigned NOT NULL,
  PRIMARY KEY (`id`)
);

CREATE TABLE `house_lease_terms` (
  `id` int(10) unsigned NOT NULL AUTO_INCREMENT,
  `house_lease_id` int(10) unsigned NOT NULL,
  `date_start` datetime NOT NULL,
  `date_end` datetime DEFAULT NULL,
  PRIMARY KEY (`id`),
  KEY `house_lease_terms_house_lease_id_foreign` (`house_lease_id`),
  CONSTRAINT `house_lease_terms_house_lease_id_foreign` FOREIGN KEY (`house_lease_id`) REFERENCES `house_leases` (`id`) ON DELETE CASCADE
);

如您所见, house_lease_terms.house_lease_id对应于特定的house_lease ,但是可以有多个具有相同house_lease_id 的行。

确定“活动”术语的规则是:

date_start <= NOW() AND date_end > NOW() OR date_end IS NULL

如果没有返回行,那么“活动”条款必须在未来,因此规则更改为:

date_start > 现在()

如果条款不在未来,我们按date_start DESC排序,因为可能返回多行,我们希望最新的date_start位于结果的顶部。 否则,我们按date_start ASC排序,因为我们希望现在最接近的date_start位于顶部。

然后我将其限制为 1 以仅获得一个结果,并且该行被视为“活动”术语。 如果没有返回结果,则没有“活动”术语。

我有一个 SQL 语句,它具有获取特定house_lease_id 的逻辑。 看起来像这样:

SELECT * FROM house_lease_terms
WHERE 
CASE 
    WHEN 
        (SELECT COUNT(*) FROM house_lease_terms WHERE date_start <= NOW() AND (date_end > NOW() OR date_end IS NULL) AND house_lease_id = 1)
    THEN 
        date_start <= NOW() AND (date_end > NOW() OR date_end IS NULL)
    ELSE
        date_start > NOW()
END
AND house_lease_id = 1
ORDER BY
    IF(
        (SELECT COUNT(*) FROM house_lease_terms WHERE date_start <= NOW() AND (date_end > NOW() OR date_end IS NULL) AND house_lease_id = 1), 
        unix_timestamp(date_start), 
        -unix_timestamp(date_start)
    ) DESC
LIMIT 1;

此语句有效,但我希望有一种更好的方法(更有效)来获取特定house_lease_id的“活动”条款(如果您知道更好的解决方案,请分享)。

现在我需要一个查询来获取所有不同house_lease_id的“活动”术语。

我不希望任何类型的自定义 MySQL 函数或存储过程来执行此操作。 我不知道从哪里开始创建这个查询。 我想我可以在某些子选择或连接中使用上面的查询,但我不确定我将如何这样做。

任何帮助将不胜感激!

带有数据的 SQLFiddle: http ://sqlfiddle.com/#!9/cab159/2/0

寻找:

SELECT *
FROM house_lease_terms
WHERE house_lease_id = 1
  AND (   (   (    date_start <= NOW() 
               AND date_end > NOW()
              ) 
           OR date_end IS NULL
          )
       OR (date_start > NOW()
          )
      )
ORDER BY CASE WHEN ((date_start <= NOW() AND date_end > NOW()) OR date_end IS NULL)
              THEN DATEDIFF(NOW(), date_start)
              ELSE DATEDIFF(date_start, NOW()) + 1000000
              END

SELECT id,
       CASE WHEN @var2=house_lease_id 
            THEN @var1:=house_lease_id 
            ELSE 1
            END row_number_in_house_lease_id,
       @var2:=house_lease_id house_lease_id,
       date_start,
       date_end,
       created_at,
       updated_at,
       deleted_at
FROM house_lease_terms, (SELECT @var1:=0, @var2:=0) variables
WHERE house_lease_id = 1
  AND (   (   (    date_start <= NOW() 
               AND date_end > NOW()
              ) 
           OR date_end IS NULL
          )
       OR (date_start > NOW()
          )
      )
ORDER BY house_lease_id,
         CASE WHEN ((date_start <= NOW() AND date_end > NOW()) OR date_end IS NULL)
              THEN DATEDIFF(NOW(), date_start)
              ELSE DATEDIFF(date_start, NOW()) + 1000000
              END

下一代。

SELECT id, house_lease_id, date_start, date_end
FROM (
SELECT id,
       CASE WHEN @var2=house_lease_id 
            THEN @var1:=@var1+1 
            ELSE @var1:=1
            END row_number_in_house_lease_id,
       @var2:=house_lease_id house_lease_id,
       date_start,
       date_end
FROM house_lease_terms, (SELECT @var1:=0, @var2:=0) variables 
WHERE (   (   (    date_start <= NOW() 
               AND date_end > NOW()
              ) 
           OR date_end IS NULL
          )
       OR (date_start > NOW()
          )
      )
ORDER BY house_lease_id,
         CASE WHEN ((date_start <= NOW() AND date_end > NOW()) OR date_end IS NULL)
              THEN DATEDIFF(NOW(), date_start)
              ELSE DATEDIFF(date_start, NOW()) + 1000000
              END
) AS cte
WHERE row_number_in_house_lease_id = 1;

小提琴

我不确定这是你认为干净的,但似乎任何方法都会导致相当复杂和混乱的查询。

根据您的原始查询:

SELECT house_lease_id, 
       SUBSTRING_INDEX(GROUP_CONCAT(id ORDER BY
                                    IF(
                                        (SELECT COUNT(*) FROM house_lease_terms AS hlt2 
                                         WHERE date_start <= NOW() 
                                         AND (date_end > NOW() OR date_end IS NULL) 
                                         AND hlt2.house_lease_id = hlt.house_lease_id), 
                                         unix_timestamp(date_start), 
                                         -unix_timestamp(date_start)
                                    ) DESC),',',1) AS id,
       SUBSTRING_INDEX(GROUP_CONCAT(date_start ORDER BY
                                    IF(
                                        (SELECT COUNT(*) FROM house_lease_terms AS hlt3 
                                         WHERE date_start <= NOW() 
                                         AND (date_end > NOW() OR date_end IS NULL) 
                                         AND hlt3.house_lease_id = hlt.house_lease_id), 
                                         unix_timestamp(date_start), 
                                         -unix_timestamp(date_start)
                                    ) DESC),',',1) AS date_start,
       SUBSTRING_INDEX(GROUP_CONCAT(date_end ORDER BY
                                    IF(
                                        (SELECT COUNT(*) FROM house_lease_terms AS hlt4 
                                         WHERE date_start <= NOW() 
                                         AND (date_end > NOW() OR date_end IS NULL) 
                                         AND hlt4.house_lease_id = hlt.house_lease_id), 
                                         unix_timestamp(date_start), 
                                         -unix_timestamp(date_start)
                                    ) DESC),',',1) AS date_end 
FROM house_lease_terms AS hlt
WHERE 
CASE 
    WHEN 
        (SELECT COUNT(*) FROM house_lease_terms AS hlt5 WHERE date_start <= NOW() 
         AND (date_end > NOW() OR date_end IS NULL) 
         AND hlt5.house_lease_id = hlt.house_lease_id)
    THEN 
        date_start <= NOW() AND (date_end > NOW() OR date_end IS NULL)
    ELSE
        date_start > NOW()
END
GROUP BY hlt.house_lease_id;

那应该工作。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM