在where子句中使用group by进行条件计数

时间:2015-05-16 11:04:06

标签: mysql sql group-by

我有一个简单的消息系统 - 将所有消息保存在一个表中。每条消息都可以(并且应该)与其他3个表中的一个相关联,这些表代表网站的某些部分。这是create table语句

CREATE TABLE `messages` (
  `id` bigint(20) unsigned NOT NULL AUTO_INCREMENT,
  `from_user_id` int(11) DEFAULT NULL,
  `to_user_id` int(11) DEFAULT NULL,
  `message` text COLLATE utf8_bin,
  `table1_id` int(11) DEFAULT NULL,
  `table2_id` int(11) DEFAULT NULL,
  `table3_id` int(11) DEFAULT NULL,
  `is_unread` tinyint(1) DEFAULT NULL,
  `date` datetime DEFAULT NULL,
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=14 DEFAULT CHARSET=utf8 COLLATE=utf8_bin;

对于每条消息,如果某列有值,则从table1_id, table2_id and table3_id开始,表示其余2为空。这是sqlfiddle结构和示例数据:http://sqlfiddle.com/#!9/b98a2/1/0

因此,table1_id, table2_id and table3_id是一种线程,我在分组时使用 - 显示消息列表。这是我的查询

SELECT 
  id,
  table1_id,
  table2_id,
  table3_id,
  message,
  from_user_id,
  to_user_id,
  COUNT(table1_id) AS t1_count,
  COUNT(table2_id) AS t2_count,
  COUNT(table3_id) AS t3_count,
  MAX(CASE WHEN to_user_id = 10 AND is_unread = 1 THEN 1 END) AS is_unread,
  COUNT(CASE WHEN to_user_id = 10 THEN 1 END) AS inbox_count
FROM
  messages
WHERE to_user_id = 10 OR from_user_id = 10
GROUP BY table1_id,
  table2_id,
  table3_id
ORDER BY id DESC

这是在sqlfiddle http://sqlfiddle.com/#!9/b98a2/2/0

当我必须显示所有消息时,此查询正常工作,但是,例如,我想只显示具有id = 10的用户的收件箱我必须检查每个线程至少有一条收到的消息的情况,所以为此我尝试应用条件AND inbox_count > 0,结果错误Unknown column 'inbox_count' in 'where clause'

我正在尝试列出类似于gmail的消息 - 显示每个帖子的总消息数(t1_count, t2_count or t3_count),这就是我无法删除OR from_user_id = 10部分的原因。

为什么它找不到该列以及如何应用该条件来显示已接收(传出)的消息列表。

1 个答案:

答案 0 :(得分:2)

除非我完全误解你的意图......如果你想用inbox_count > 0进行过滤,那么我想你想添加一个

HAVING COUNT(CASE WHEN to_user_id = 10 THEN 1 END) > 0

group by子句之后。这将删除没有to_user = 10的任何消息的“线程”。

See this fiddle和示例。