mysql必知必会——GROUP BY和HAVING

mysql必知必会——GROUP BY和HAVING

创建表结构

create table `employ_info` (
   `id` int(11) NOT NULL AUTO_INCREMENT,
   `name` char(20) NOT NULL DEFAULT '',
   `dept` char(20) NOT NULL DEFAULT '',
   `salary` varchar(255) NOT NULL DEFAULT '',
   `edlevel` int(11) NOT NULL DEFAULT 0,
   `hiredate` varchar(255) NOT NULL DEFAULT '',
   PRIMARY KEY(`id`)
) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8;

添加表内容

INSERT INTO `employ_info` VALUES(1,'张三','开发部','2000','3','2009-10-11'),
(2,'李四','开发部','2500','3','2009-10-11'),
(3,'王五','设计部','2600','5','2010-10-02'),
(4,'王六','设计部','2300','4','2010-10-03'),
(5,'马七','设计部','2100','4','2010-10-05'),
(6,'赵八','销售部','3000','5','2010-10-07'),
(7,'钱九','销售部','3100','7','2010-10-07'),
(8,'孙十','销售部','3500','7','2010-10-06');

GROUP BY语法可以根据给定数据列的每个成员对查询结果进行分组统计,最终得到一个分组汇总表。

select子句中的列名必须为分组列或列函数,列函数对于group by子句定义的每个组返回一个结果。

某个员工信息表结构和数据如下:

mysql> select * from employ_info;
+----+--------+-----------+--------+---------+------------+
| id | name   | dept      | salary | edlevel | hiredate   |
+----+--------+-----------+--------+---------+------------+
|  1 | 张三   | 开发部    | 2000   |       3 | 2009-10-11 |
|  2 | 李四   | 开发部    | 2500   |       3 | 2009-10-11 |
|  3 | 王五   | 设计部    | 2600   |       5 | 2010-10-02 |
|  4 | 王六   | 设计部    | 2300   |       4 | 2010-10-03 |
|  5 | 马七   | 设计部    | 2100   |       4 | 2010-10-05 |
|  6 | 赵八   | 销售部    | 3000   |       5 | 2010-10-07 |
|  7 | 钱九   | 销售部    | 3100   |       7 | 2010-10-07 |
|  8 | 孙十   | 销售部    | 3500   |       7 | 2010-10-06 |
+----+--------+-----------+--------+---------+------------+
8 rows in set (0.07 sec)

我想列出每个部门最高薪水的结果,sql语句如下:

mysql> select dept,max(salary) AS MAXIMUM from employ_info group by dept;
+-----------+---------+
| dept      | MAXIMUM |
+-----------+---------+
| 开发部    | 2500    |
| 设计部    | 2600    |
| 销售部    | 3500    |
+-----------+---------+
3 rows in set (0.37 sec)

解释一下这个结果:

1、 满足“SELECT子句中的列名必须为分组列或列函数”,因为SELECT有group by中包含的列dept;

2、“列函数对于group by子句定义的每个组各返回一个结果”,根据部门分组,对每个部门返回一个结果,就是每个部门的最高薪水。

<font color=#00ffff>将where子句与group by子句一起使用</font>

分组查询可以在形成组和计算列函数之前具有消除非限定行的标准where子句。<font color=red>必须在group by子句之前指定where子句</font>


例如,查询公司2010年入职的各个部门每个级别里的最高薪水

mysql> select dept,edlevel,MAX(salary) AS MAXIMUM from employ_info group by dept,edlevel;
+-----------+---------+---------+
| dept      | edlevel | MAXIMUM |
+-----------+---------+---------+
| 开发部    |       3 | 2500    |
| 设计部    |       4 | 2300    |
| 设计部    |       5 | 2600    |
| 销售部    |       5 | 3000    |
| 销售部    |       7 | 3500    |
+-----------+---------+---------+
5 rows in set (0.00 sec)

mysql> select dept,edlevel,MAX(salary) AS MAXIMUM from employ_info where hiredate='2010-01-01' group by dept,edlevel;
Empty set (0.00 sec)

mysql> select dept,edlevel,MAX(salary) AS MAXIMUM from employ_info where hiredate='2010-01-07' group by dept,edlevel;
Empty set (0.00 sec)

<font color=red>在SELECT语句中指定的每个列名也在GROUP BY子句中提到,未在这两个地方提到的列名将产生错误。GROUP BY子句对dept和edlevel的每个唯一组合各返回一行。</font>

<font color=#00ffff>GROUP BY子句之后使用Having子句</font>

可应用限定条件进行分组,以便系统仅对满足条件的组返回结果。因此,在GROUP BY子句后面包含了一个HAVING子句。HAVING类似于WHERE(唯一的差别是WHERE过滤行,HAVING过滤组)AVING支持所有WHERE操作符。

例如,查找雇员数超过2个的部门的最高和最低薪水:

mysql> select dept,MAX(salary) as MAXIMUM,MIN(salary) as MINIMUM from employ_info group by dept having count(*)>2 order by dept;
+-----------+---------+---------+
| dept      | MAXIMUM | MINIMUM |
+-----------+---------+---------+
| 设计部    | 2600    | 2100    |
| 销售部    | 3500    | 3000    |
+-----------+---------+---------+
2 rows in set (0.09 sec)

例如,查找雇员平均工资大于3000的部门的最高薪水和最低薪水:

mysql> select dept,max(salary) as MAXIMUM,MIN(salary) as MINIMUM from employ_info group by dept having avg(salary)>3000 order by dept;
+-----------+---------+---------+
| dept      | MAXIMUM | MINIMUM |
+-----------+---------+---------+
| 销售部    | 3500    | 3000    |
+-----------+---------+---------+
1 row in set (0.00 sec)
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 195,980评论 5 462
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 82,422评论 2 373
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 143,130评论 0 325
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,553评论 1 267
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,408评论 5 358
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,326评论 1 273
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,720评论 3 386
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,373评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,678评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,722评论 2 312
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,486评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,335评论 3 313
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,738评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,009评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,283评论 1 251
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,692评论 2 342
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,893评论 2 335

推荐阅读更多精彩内容