mysql group by笔试题_面试笔试常考的mysql 数据库操作group by

本文详细介绍了SQL中的GROUP BY语句及其在数据库查询中的应用,包括如何根据部门获取最高分、筛选特定条件(如性别)以及利用HAVING子句设置分组后的过滤条件。此外,还讨论了DISTINCT与GROUP BY的区别和配合使用的方法,展示了在解决实际问题时如何灵活运用SQL语句。
摘要由CSDN通过智能技术生成

IT 面试中,数据库的相关问题基本上属于必考问题,而其中关于sql语句也是经常考察的一个重要知识点。

下面介绍下sql语句中一个比较重要的操作group by,他的重要行一方面体现在他的理解困难度,一方面体现应用中的长见性。

首先,给出一个studnet学生表:

CREATE TABLE `student` (

`id` int(11) NOT NULL AUTO_INCREMENT,

`name` varchar(30) DEFAULT NULL,

`sex` tinyint(1) DEFAULT '0',

`score` int(10) NOT NULL,

`dept` varchar(10) DEFAULT NULL,

PRIMARY KEY (`id`)

) ENGINE=MyISAM AUTO_INCREMENT=8 DEFAULT CHARSET=utf8

添加一些测试数据:

mysql> select * from student where id<10;

+----+------+------+-------+---------+

| id | name | sex | score | dept |

+----+------+------+-------+---------+

| 1 | a | 1 | 90 | dev |

| 2 | b | 1 | 90 | dev |

| 3 | b | 0 | 88 | design |

| 4 | c | 0 | 60 | sales |

| 5 | c | 0 | 89 | sales |

| 6 | d | 1 | 100 | product |

+----+------+------+-------+---------+

给出需求,写出sql:

给出各个部门最高学生的分数。

要想得到各个部门学生,首先就要分组,按照部门把他们分组,然后在各个部门中找到分数最高的就可以了。

所以sql语句为:

mysql> select *, max(score) as max from student group by dept order by name;

+----+------+------+-------+---------+------+

| id | name | sex | score | dept | max |

+----+------+------+-------+---------+------+

| 1 | a | 1 | 90 | dev | 90 |

| 3 | b | 0 | 88 | design | 88 |

| 4 | c | 0 | 60 | sales | 89 |

| 6 | d | 1 | 100 | product | 100 |

+----+------+------+-------+---------+------+

4 rows in set (0.00 sec)

这只是个简单的例子,我们可以再把这个例子复杂化,比如分数最高的必须是女生,即sex列值必须为1才挑选出,这时的sql语句应该为:

mysql> select *,max(score) as max from student group by dept having sex='1' order by name;

+----+------+------+-------+---------+------+

| id | name | sex | score | dept | max |

+----+------+------+-------+---------+------+

| 1 | a | 1 | 90 | dev | 90 |

| 6 | d | 1 | 100 | product | 100 |

+----+------+------+-------+---------+------+

2 rows in set (0.46 sec)

这里我们没有用where语句而是用了having,这里简单说明一下,因为我们的条件是在分组后进行的,其实分组前挑选出sex=‘1‘,然后再按照dept部门分组,也是可行的,这里就要看题目是怎么要求的:

mysql> select *,max(score) as max from student where sex='1' group by dept order by name;

+----+------+------+-------+---------+------+

| id | name | sex | score | dept | max |

+----+------+------+-------+---------+------+

| 1 | a | 1 | 90 | dev | 90 |

| 6 | d | 1 | 100 | product | 100 |

+----+------+------+-------+---------+------+

2 rows in set (0.05 sec)

查询出的结果时一致的,如果把选择条件改为必须部门所有人的分数之和大于150才能把分数最高的部门的人列出来,这里就必须使用having了,因为 having 里面可以使用聚合函数sum,并且也必须分完组我们才能得到这个组的总分数,才能比较是否该值大于150:

mysql> select *,max(score) as max from student group by dept having sum(score)>150 order by name;

+----+------+------+-------+---------+------+

| id | name | sex | score | dept | max |

+----+------+------+-------+---------+------+

| 1 | a | 1 | 90 | dev | 90 |

| 6 | d | 1 | 100 | product | 100 |

+----+------+------+-------+---------+------+

2 rows in set (0.00 sec)

额外增加一个例子,比如我要选出不重复的部门,我们可以使用

mysql> select distinct dept from student;

+---------+

| dept |

+---------+

| dev |

| design |

| sales |

| product |

+---------+

4 rows in set (0.02 sec)

但是如果我们还要列出他的id等一些其他信息,我们如果这样:

mysql> select name,distinct dept from student;

ERROR 1064 (42000): You have an error in your SQL syntax; check the manual that corresponds to your MySQL server version for the right syntax to use near 'distinct dept from student' at line 1

这是不行的,因为distinct只能放到开始位置,如果:

mysql> select distinct dept,name from student;

+---------+------+

| dept | name |

+---------+------+

| dev | a |

| dev | b |

| design | b |

| sales | c |

| product | d |

| product | m |

+---------+------+

6 rows in set (0.00 sec)

为什么没有达到预期的效果,因为distinct 作用到了2个字段上,这时,我们就需要groub by 出场了。

mysql> select dept,name from student group by dept;

+---------+------+

| dept | name |

+---------+------+

| design | b |

| dev | a |

| product | d |

| sales | c |

+---------+------+

4 rows in set (0.00 sec)

按照dept分组,自然就达到去重的目的了。所以有时候如果我们碰到了一个问题很难解决,比如用distinct去重,并带上其他列值,我们就需要尝试换个思路,可能答案自然就找到了。

原文:http://blog.csdn.net/fujiafeihudui/article/details/38402467

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值