iis服务器助手广告广告
返回顶部
首页 > 资讯 > 数据库 >mysql组内排序取最大值
  • 176
分享到

mysql组内排序取最大值

2024-04-02 19:04:59 176人浏览 薄情痞子
摘要

最近业务反馈一个查询异常的问题,需要DBA对查询结果异常给出解释,并帮助他们解决该问题。问题本质是一个组内排序取最大值的问题,根据业务需求,我构建了测试用例测试用例--建表 create tabl

最近业务反馈一个查询异常的问题,需要DBA对查询结果异常给出解释,并帮助他们解决该问题。问题本质是一个组内排序取最大值的问题,根据业务需求,我构建了测试用例

测试用例

--建表
create table testorder
(id int not null,
no int not null,
name char(10) not null,
primary key(id)
)engine=innodb;
--写入数据
insert into testorder values (1,1,'Mike'),(2,2,'John'),(3,3,'wyett'),(4,4,'Herry'),(5,5,'Mike'),(6,1,'John'),(7,2,'John'),(8,1,'Mike'),(9,1,'Mike');
--查询1
select * from testorder;
+----+----+-------+
| id | no | name  |
+----+----+-------+
|  1 |  1 | Mike  |
|  2 |  2 | John  |
|  3 |  3 | wyett |
|  4 |  4 | Herry |
|  5 |  5 | Mike  |
|  6 |  1 | John  |
|  7 |  2 | John  |
|  8 |  1 | Mike  |
|  9 |  1 | Mike  |
+----+----+-------+
--查询2
select * from testorder order by no desc;
+----+----+-------+
| id | no | name  |
+----+----+-------+
|  5 |  5 | Mike  |
|  4 |  4 | Herry |
|  3 |  3 | wyett |
|  2 |  2 | John  |
|  7 |  2 | John  |
|  1 |  1 | Mike  |
|  6 |  1 | John  |
|  8 |  1 | Mike  |
|  9 |  1 | Mike  |
+----+----+-------+
--查询3select * from (select id,no,name from testorder order by no desc)a group by a.name;

查询3这条sql是我们需要讨论的内容,也是业务线为实现组内排序取最大值所采用的SQL。标准的程序员反馈问题方式:XXX时间点之前查询时正常的,这之后突然就不正常了,你们DBA是不是做什么改动了?我把数据恢复到自己的测试机,返回值也是正常的。暂且不去管姿势是否正确,对这条SQL的分析,我们其实可以看出:(1)程序员期待group by执行结果是按照临时表a的数据顺序来取值;(2)程序员未考虑版本因素,数据量变化的因素;为此,我构建了上面的测试用例。

测试

在不同版本的Mysql来进行测试:发现在Percona 5.5,Percona 5.1,mysql 5.6关闭sql_mode= ONLY_FULL_GROUP_BY,MySQL5.1等版本下,返回值确如程序员期待的顺序,按照order by no desc的顺序,相同name返回no值最大的数据;

+----+----+-------+
| id | no | name  |
+----+----+-------+
|  4 |  4 | Herry |
|  2 |  2 | John  |
|  5 |  5 | Mike  |
|  3 |  3 | wyett |
+----+----+-------+

在mysql5.7,关闭sql_mode= ONLY_FULL_GROUP_BY和mariadb 10.*版本中,相同的name值,返回则是取了最早写入的数据行,忽略了order by no desc,按照数据的逻辑存储顺序来返回;

+----+----+-------+
| id | no | name  |
+----+----+-------+
|  4 |  4 | Herry |
|  2 |  2 | John  |
|  1 |  1 | Mike  |
|  3 |  3 | wyett |
+----+----+-------+

其实在这里,SQL等价于select id,no,name from testorder group by name。
这里我们看出不同版本的返回值是不同的,先搁置数据量的变化引起执行结果不同的讨论,因为数据量大小很难测试。

官方文档

对上面的测试结果,在官方文档上,有如下的参考

If ONLY_FULL_GROUP_BY is disabled...In this case, the server is free to choose any value from each group, 
so unless they are the same, the values chosen are indeterminate, which is probably not what you want. 
Furthermore, the selection of values from each group cannot be influenced by adding an ORDER BY clause. 
Result set sorting occurs after values have been chosen, and ORDER BY does not affect which value within 
each group the server chooses.

ONLY_FULL_GROUP_BY这个SQL_MODE出在mysql5.6(mariadb 10.0)时被引入,但本文讨论的内容和它无关,具体可以自己查看文档,这里不做讨论。在5.6,5.5的官方文档有相同的内容,Mariadb也有类似的解释

 If you select a non-grouped column or a value computed from a non-grouped column, it is undefined 
which row the returned value is taken from. This is not permitted if the ONLY_FULL_GROUP_BY SQL_MODE is used.

并且,对from后的subquery子表中的order by也给出了解释

 A query such as

SELECT field1, field2 FROM ( SELECT field1, field2 FROM table1 ORDER BY field2 ) alias
returns a result set that is not necessarily ordered by field2. This is not a bug.

A "table" (and subquery in the FROM clause too) is - according to the SQL standard - an unordered set of rows. 
Rows in a table (or in a subquery in the FROM clause) do not come in any specific order.

好了,有了这些解释,问题很明朗:

  • 在from 后的subquery中的order by会被忽略

  • group by cloumn返回的行是无序的

因此,业务获得的正确的返回值也是误打误撞。

解决办法

那么这个问题该怎么解决?

在网上有一些SQL,很明显不满足需求,在这里做一下展示,希望同学们避免被误导:

错误SQL集合

select id,sbustring(GROUP_CONCAT(distinct no order by no desc separator ''),'',1),name from testorder group by name;
--通过添加索引来影响返回的结果集顺序
alter table testorder add index idx_no_name(no desc, name);
--结果证明即使如此,desc也不会被正确执行;
--我司程序员的写法
select * from (select id,no,name from testorder order by no desc)a group by a.name
select id,max(no),name from testorder group by name

我们可以这样写,虽然效率不高

select a.id,a.no,a.name 
from testorder a 
inner join (select max(no) no,name 
            from testorder 
	    group by name) b on a.no=b.no and a.name=b.name 
group by name,no

或者这样

select a.id,a.no,a.name 
from testorder a 
group by a.name,a.no 
having a.no=(select max(no) from testorder where name=a.name)

 


您可能感兴趣的文档:

--结束END--

本文标题: mysql组内排序取最大值

本文链接: https://www.lsjlt.com/news/37373.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • SQL如何实现组内排序取最大值
    这篇文章主要介绍了SQL如何实现组内排序取最大值,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。 测试用例--建表 create ...
    99+
    2024-04-02
  • java lambda 数据库 分组排序取最大一条记录(组内排序取最大一条)
    1. 说明   最近遇到一个问题,一个记录表,需要批量根据某一个特定字段,获取这个字段最大记录,也就是需要先分组再组内排序,取组内最大一条数据 2. 解决方案   组内排序获取最大一条记录,本文探讨的...
    99+
    2023-09-09
    数据库 java mysql
  • mysql分组排序取组内第一的数据行获取分组后,组内排名第一或最后的数据行。
    前言: group by函数后取到的是分组中的第一条数据,但是我们有时候需要取出各分组的最新一条,该怎么实现呢? 本文提供两种实现方式。 一、准备数据 DROP TABLE IF EXISTS `tb_dept`;CREATE TA...
    99+
    2023-09-08
    数据库 sql mysql
  • oracle中排序后取第一条数据怎么取最大值
    在 oracle 中,获取排序后第一条数据的最大值,可以使用 order by 子句和 limit 子句:select column_name:选择要取最大值的列from ta...
    99+
    2024-05-08
    oracle
  • 在 MySQL 中获取最小值和最大值
    我们需要使用 MAX(columnName) 来查找列中的最大值,而使用 MIN(columnName) 来查找列中的最大值。假设以下是语法查找特定列中的最高值和最低值 -mysql> SELECT @min_val:=MIN...
    99+
    2023-10-22
  • sparkdataframe全局排序id与分组后保留最大值行
    目录正文(1) 得到 spark dataframe 全局排序ID(2)分组后保留最大值行正文 作为一个算法工程师,日常学习和工作中,不光要 训练模型关注效果 ,更多的 时间 是在...
    99+
    2023-02-09
    spark dataframe全局排序 spark dataframe
  • 列表内数字组合最大值
    第一种import itertoolslt = [4, 40, 45, 6, 9, 3, 5, 2, 8]lt2 = map(str, lt)it = itertools.permutations(lt2,len(lt))# for i ...
    99+
    2023-01-31
    组合 最大值 数字
  • JavaScript如何从数组中获取最大值和最小值
    小编给大家分享一下JavaScript如何从数组中获取最大值和最小值,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!从数组中获取最...
    99+
    2024-04-02
  • mysql日期如何获取最大值
    这篇文章主要介绍了mysql日期如何获取最大值的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇mysql日期如何获取最大值文章都会有所收获,下面我们一起来看看吧。操作步骤:1、首...
    99+
    2024-04-02
  • mysql日期怎么获取最大值
    mysql日期获取最大值的方法:1、查询数据库表中日期字段和信息。2、调用max()获取最大值。具体操作步骤:首先输入语句查询mysql表中的数据,得到日期的字段和信息,代码如下:select * from&n...
    99+
    2024-04-02
  • MySQL如何实现组内排序
    MySQL如何实现组内排序?这个问题可能是我们日常学习或工作经常见到的。希望通过这个问题能让你收获颇深。下面是小编给大家带来的参考内容,让我们一起来看看吧!假设表格为student, 数据如下:  我们要在...
    99+
    2024-04-02
  • mysql怎么取最大值行数据
    要取最大值行数据,可以使用以下SQL语句: SELECT * FROM your_table WHERE your_...
    99+
    2024-04-02
  • python groupby函数实现分组选取最大值与最小值
    现在需要将course分组,然后选择出每一组里面的最大值和最小值,并保留下来 实现下面数据结果: 直接使用groupby函数,不能直接达到此效果,需要在groupby函数上添加a...
    99+
    2024-04-02
  • MySQL中如何使用GROUP BY分组取字段最大值
    小编给大家分享一下MySQL中如何使用GROUP BY分组取字段最大值,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧! ...
    99+
    2024-04-02
  • java数组算法例题代码详解(冒泡排序,选择排序,找最大值、最小值,添加、删除元素等)
    目录数组算法例题1.数组逆序2.找出数组中最大值所在下标位置3.找出数组中指定元素第一次出现的下标位置4.在数组中找出指定下标对应的元素5.找出指定元素在数组中最后一次出现位置6.找...
    99+
    2024-04-02
  • MYSQL排序将NULL值放在最后表示
    通常情况mysql 升序排序 默认会把NULL放在前面,因为在排序过程中默认会将NULL当作最小值处理,降序的情况就会排在最后面。 有些情况下在升序中要求将NULL在最后,进行如下处理: 假设表查询并通过Sort_No字段进行升序排序,表中...
    99+
    2023-09-06
    mysql 数据库
  • Java怎么获取数组中的最大值
    要获取数组中的最大值,可以通过以下方法实现: 使用循环遍历数组,逐个比较数组中的元素,找到最大值。 public static ...
    99+
    2024-03-07
    Java
  • mysql怎么获取日期最大的值
    在mysql中获取日期最大值的方法首先,在命令行中启动mysql服务;service mysql start  MySQL服务启动后,在命令行中输入mysql的用户名和密码登录到mysql;mysql -u root -p登录到mysql后...
    99+
    2024-04-02
  • java怎么获取数组中最大的值
    你可以使用循环遍历数组,然后比较每个元素的值来找到最大值。以下是一个使用Java的示例代码:```public class Main...
    99+
    2023-08-16
    java
  • r语言怎么获取数组中的最大值和最小值
    在R语言中,可以使用max()函数和min()函数来获取数组中的最大值和最小值。 例如,假设有一个包含一组数字的数组x,要获取该数组...
    99+
    2024-03-06
    r语言
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作