Mysql联合查询UNION和Order by同时使用报错问题的解决办法

 更新时间:2014年04月12日 09:20:12   作者:  
很多朋友刚使用联合查询UNION的时候常常会理所当然的将联合查询理解为把没一个子查询的结果集组合成一个大的结果集

因此,常常出现这样的错误

复制代码 代码如下:

select * from [IND] where INDID>10
union
select * from [IND] where INDID<9

目前为止,还没有出现问

之后,也许有人会用到类似的查询

复制代码 代码如下:

select * from [IND] where INDID>10 order by INDID desc
union
select * from [IND] where INDID<9 order by INDID desc

此时就出现问题了,数据库报错。问题就出在order by上

为什么呢?难道UNION和ORDER BY 不能同时存在?

union和 order by 当然是可以同时存在的

但是在使用union的时候,联合查询不仅仅是将数据集合合并
他并不是将每个子查询一个一个查询出来后联接在一起,数据库是将整段查询语句理解之后统一查询得到的是整个的数据集合
另外order by在一个数据集合查询里也只能出现一次并且出现在最后。
因此,在联合查询里,order by 要写在最后一个子查询之后,并且,该排序是对整个联合查询出来的结果集排序的,并不是只对最后一个子查询排序

复制代码 代码如下:
select * from [IND] where INDID>10
union
select * from [IND] where INDID<9 order by INDID desc

这样就可以对我们联合查询出来的结果集进行整体排序,而不是只对最后一个子查询的结果集排序。

再做一个试验来更充分的说明这个问题

创建一个这样的查询

复制代码 代码如下:

select * from [IND] where INDID=4

union

select * from [IND] where INDID=2
union
select * from [IND] where INDID=1
union
select * from [IND] where INDID=5
union
select * from [IND] where INDID=3


INDID是主键,在创建数据的时候,数据库里的顺序是12345


如果联合查询只是但存的将查询结果联接在一起,那么我们得到的结果应该是:42153

但是,实际上得到的结果是和数据库里数据排列的顺序一样的 12345

因此,可以得出结论,联合查询的结果是整个查询完成后得出的,而不是将子查询挨个完成后拼接的。

复制代码 代码如下:

select * from [IND] where INDID=4

union

select * from [IND] where INDID=2
union
select * from [IND] where INDID=1
union
select * from [IND] where INDID=5
union
select * from [IND] where INDID=3

order by INDID ASC/DESC


这样就可以对整个联合结果集进行排寻了。


另外关于TOP?

如果是这样,在普通的查询中,TOP是在ORDER BY之后执行的,那么

复制代码 代码如下:

select TOP 2 * from [IND] where INDID=4

union

select * from [IND] where INDID=2
union
select * from [IND] where INDID=1
union
select * from [IND] where INDID=5
union
select * from [IND] where INDID=3

order by INDID


这样是不是可以得到整个结果集排序后的最前面两条数据呢?

答案是不可以。

虽然说在单句的查询中,TOP是在ORDER BY 之后执行,但是在联合查询中,这样写,TOP的作用域是在子查询里,因此TOP并没有对联合查询的结果集筛选,而只对它所写在的那条子查询里进行筛选,这就像是子查询里的WHERE语句一样,类似这样的筛选作用范围都是在子查询,不像ORDER BY 作用在整个联合查询。


那么如何对联合查询进行 截取置顶N条数据的筛选呢? 很简单

用 rowcount

比起TOP来说,rowcount作为结果集截取置顶更加规范些,毕竟不是依靠查询语句,而是直接设置查询语句获得结果集的数目。

复制代码 代码如下:

set rowcount 2

select * from [IND] where INDID=4

union
select * from [IND] where INDID=1
union
select * from [IND] where INDID=2

union
select * from [IND] where INDID=3

order by INDID ASC


形如以上查询语句。我们就可以做到对联合查询排序,并获得最上的两条数据了。


既然能利用order by 排序 和 并用rowcount截取集和数量,那么自然联合查询分页等其他应用也不在话下了

相关文章

  • 图文详解mysql5.7安装教程

    图文详解mysql5.7安装教程

    这篇文章主要以图文结合的方式为大家详细介绍了mysql5.7安装教程的相关资料,需要的朋友可以参考下
    2016-05-05
  • SQL行列转换超详细四种方法详解

    SQL行列转换超详细四种方法详解

    在数据分析的面试中SQL问题基本上是必问的,其中SQL行列转换的问题出镜率极其高,重要性也是不言而喻,下面这篇文章主要给大家介绍了关于SQL行列转换超详细四种方法的相关资料,需要的朋友可以参考下
    2022-12-12
  • 安全地关闭MySQL服务的教程

    安全地关闭MySQL服务的教程

    这篇文章主要介绍了安全地关闭MySQL服务的教程,包括使用kill指令时的一些注意点,需要的朋友可以参考下
    2015-11-11
  • 小白安装登录mysql-8.0.19-winx64的教程图解(新手必看)

    小白安装登录mysql-8.0.19-winx64的教程图解(新手必看)

    这篇文章主要介绍了安装登录mysql-8.0.19-winx64的教程图解,非常适合新手学习参考,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-03-03
  • MySQL数据库10秒内插入百万条数据的实现

    MySQL数据库10秒内插入百万条数据的实现

    假设现在我们要向mysql插入500万条数据,如何实现高效快速的插入进去?本文就详细的介绍一下,感兴趣的可以了解一下
    2021-10-10
  • MySQL数据库之索引详解

    MySQL数据库之索引详解

    大家好,本篇文章主要讲的是MySQL数据库之索引详解,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下,方便下次浏览
    2021-12-12
  • 基于Mysql的IP处理函数inet_aton()与inet_ntoa()的深入分析

    基于Mysql的IP处理函数inet_aton()与inet_ntoa()的深入分析

    本篇文章是对Mysql的IP处理函数inet_aton()与inet_ntoa()进行了详细的分析介绍,需要的朋友参考下
    2013-06-06
  • 关于MySQL中Update使用方法举例

    关于MySQL中Update使用方法举例

    这篇文章主要给大家介绍了关于MySQL中Update使用方法的相关资料,更新数据是使用数据库时最重要的任务之一,在本教程中您将学习如何使用MySQL UPDATE语句来更新表中的数据,需要的朋友可以参考下
    2023-11-11
  • Windows中MySQL root用户忘记密码解决方案

    Windows中MySQL root用户忘记密码解决方案

    在实际应用中,经常会出现忘记mysql管理员用户root的密码的情况出现,那么我们如何来设置一个新密码从而登录数据库呢,下面我们来探讨下
    2014-07-07
  • Mysql Sql语句注释大全

    Mysql Sql语句注释大全

    这篇文章主要介绍了Mysql Sql语句注释大全,需要的朋友可以参考下
    2017-07-07

最新评论