【译注:此文为翻译,由于本人水平所限,疏漏在所难免,欢迎探讨指正】

原文链接:传送门

什么是排序函数(Ranking Functions)?

排序函数基于一组记录的集合返回一个排序值。一个排序值其实就是一个数字,典型的它都是从1开始并且对于每一个新的排序值它都是按1递增的。根据排序函数的不同,对于数据集中的每一行来说,返回的排序值有可能是唯一的,或者一些数据行会具有一样的排序值。在接下来的章节中,我将浏览下这些不的排序函数,以及它们是如何使用的。

使用排序函数(Ranking Function)的例子

在一个分区中每行的排序数字都是从1开始递增的。在一个排序函数中,一个“分区”指的其实就是一组数据行,它们对于指定的分区列具有相同的值(因而把它们归为一个分区)。如果对于一个分区的两行它们的排序列(在ORDER BY 中指定的列)具有相同的值,那么它们两个都会得到相同的排序值。为了更好的理解如何 使用Rank函数,让我们看看它的句法:

RANK ( ) OVER ( [ PARTITION BY <partition_column> ] ORDER BY <order_by_column> )

其中:

  • <partition_column>:定义了一个或者多个列名,它们将用来对数据进行分区;
  • <order by column>:定义了一个或者多个列表,它们将用来对各个分区的输出进行排序;

注意:

PARTITION BY子句是可选的。如果没有使用  PARTITION BY子句,那么数据会基于一个分区进行排序。如果你在Rank函数中指定了PARTITION BY子句,那么对于数据集中的每个分区排序值都会被重置为1的。

既然现在你已经了解了Rank函数是做什么的,以及它的句法,那么我会运行几个Rank函数的例子。我所有的例子都会使用AdventureWorks2012数据库。如果你想跟着我的示例,那么你可以从下列位置下载 AdventureWorks2012数据库:http://msftdbprodsamples.codeplex.com/releases/view/93587

对于我使用Rank函数的第一个例子,让我来运行下面的代码:

USE AdventureWorks2012;
GO
SELECT PostalCode, StateProvinceID,
RANK() OVER
(ORDER BY PostalCode ASC) AS RankingValue
FROM Person.Address
WHERE StateProvinceID IN (23,46);

列表1:简单的Rank函数的例子

当我运行列表1的代码,我得到了结果1的输出:

PostalCode      StateProvinceID RankingValue
--------------- --------------- --------------------
03064 46 1
03064 46 1
03106 46 3
03276 46 4
03865 46 5
83301 23 6
83402 23 7
83501 23 8
83702 23 9
83864 23 10

结果1: 当运行列表1的代码产生的输出

如果你查看结果1的输出,你可以看到由Rank函数产生的值在RankingValue列中。在这个例子中,我是基于PostalCode列进行排序的。每一个唯一的PostalCode值都会得到一个不同的排序值。如果你查看输出的结果行,对于PostalCode 03064,你将会看到两行,其中每一行都有一个排序值1。因为有两个PostalCode为03064的数据行,排序值2便被跳过了。对于PostalCode 03106,其排序值便会为3。剩下的RankingValue值会按次序进行分配,因为它们的PostalCode值都是唯一的。

因为Rank函数的PARTITION BY子句没有被用在列表1中,整个数据集被认为是一个单独的分区。如果我想对于每一个唯一的StateProvinceID值来重新开始我的RankingValue值,那么我必须要做的所有的事情便是基于StateProvinceID对我的结果进行分区,在列表2中我以PostalCode进行排序并以StateProvinceID进行分区。

USE AdventureWorks2012;
GO
SELECT PostalCode, StateProvinceID,
RANK() OVER
(PARTITION BY StateProvinceID
ORDER BY PostalCode ASC) AS RankingValue
FROM Person.Address
WHERE StateProvinceID IN (23,46);

列表2:使用PARTITION BY子句

当我运行列表2的代码,我得到了结果2的输出。

PostalCode      StateProvinceID RankingValue
--------------- --------------- --------------------
83301 23 1
83402 23 2
83501 23 3
83702 23 4
83864 23 5
03064 46 1
03064 46 1
03106 46 3
03276 46 4
03865 46 5

结果2:当运行列表2的输出

在列表2的输出有两个分区。一个分区包含了所有的StateProvinceID值为23的PostalCode值,第二个分区包含了StateProvinceID为46的PostalCode值。注意对每一个分区来说RankingValue都是从1开始的。

使用稠密排序(DENSE RANK)函数的例子

当我对每个重复的PostalCode值运行Rank函数,我的输出便会跳过一个RankingValue值。

通过使用DENSE RANK函数,我会生成一个不会跳过任何值的排序值。DENSE RANK函数具有如下的句法:

DENSE_RANK ( ) OVER ( [ PARTIION BY <partition_column> ] ORDER BY <order_by_column> )

其中:

  • <partition_column>:定义了一个或者多个列名,其用来对数据进行分区。
  • <order_by_column>: 定义了一个或者多个列名,其用来对各个分区的输出进行排序。

在句法上RANK 函数和 DENSE RANK函数唯一的不同其实就是函数名的不同而已。

为了浏览DENSE RANK函数让我来运行列表3的代码:

USE AdventureWorks2012;
GO
SELECT PostalCode, StateProvinceID,
DENSE_RANK() OVER
(PARTITION BY StateProvinceID
ORDER BY PostalCode ASC) AS RankingValue
FROM Person.Address
WHERE StateProvinceID IN (23,46);

列表3:使用DENSE_RANK

当我运行列表3的代码,我得到了结果3的输出:

PostalCode      StateProvinceID RankingValue
--------------- --------------- --------------------
83301 23 1
83402 23 2
83501 23 3
83702 23 4
83864 23 5
03064 46 1
03064 46 1
03106 46 2
03276 46 3
03865 46 4

结果3:运行列表3产生的输出

通过查看结果3的输出你会看到PostalCode为03064的数据行具有相同的RankingValue值。但是下一个PostalCode具有一个排序值2而不是3。记住在结果2中RANK函数对于这个相同的重复PostalCode其跳过了一个RankingValue值。使用DENSE_RANK函数,当遇到一个重复的PostalCode值时,它不会跳过一个RankingValue值。相反的,甚至当遇到重复的排序行值时,它会保证所有的RankingValue值都是连续的。

(To be continued...)

【译】高级T-SQL进阶系列 (七)【上篇】:使用排序函数对数据进行排序的更多相关文章

  1. 【译】高级T-SQL进阶系列 (七)【下篇】:使用排序函数对数据进行排序

    此文为翻译,由于本人水平有限,疏漏在所难免,欢迎探讨指正. 原文链接:传送门. 使用NTILE函数的示例 NTILE函数将一组记录分割为几个组.其返回的分组数是由一个整形表达式指定的.如下你会找到NT ...

  2. Bing Maps进阶系列七:Bing Maps功能导航菜单华丽的变身

    Bing Maps进阶系列七:Bing Maps功能导航菜单华丽的变身 Bing Maps Silverlight Control所提供的功能导航是非常强大的,在设计上对扩展的支持非常好,提供了许多用 ...

  3. 深入理解javascript函数进阶系列第一篇——高阶函数

    前面的话 前面的函数系列中介绍了函数的基础用法.从本文开始,将介绍javascript函数进阶系列,本文将详细介绍高阶函数 定义 高阶函数(higher-order function)指操作函数的函数 ...

  4. 【SQL必知必会笔记(2)】检索数据、排序检索数据

    上个笔记中介绍了一些关于数据库.SQL的基础知识,并且创建我们后续练习所需的数据库.表以及表之间的关系,从本文开始进入我们的正题:SQL语句的练习. 文章目录 1.检索数据(SELECT语句) 1.1 ...

  5. SQL进阶系列之7用SQL进行集合运算

    写在前面 集合论是SQL语言的根基,因为这种特性,SQL也被称为面向集合语言 导入篇:集合运算的几个注意事项 注意事项1:SQL能操作具有重复行的集合(multiset.bag),可以通过可选项ALL ...

  6. Linq To Sql进阶系列(六)用object的动态查询与保存log篇

    动态的生成sql语句,根据不同的条件构造不同的where字句,是拼接sql 字符串的好处.而Linq的推出,是为了弥补编程中的 Data != Object 的问题.我们又该如何实现用object的动 ...

  7. SQL进阶系列之12SQL编程方法

    写在前面 KISS -- keep it sweet and simple 表的设计 注意命名的意义 英文字母 + 阿拉伯数字 + 下划线"_" 属性和列 编程的方针 写注释 注意 ...

  8. SQL进阶系列之10HAVING子句又回来了

    写在前面 HAVING子句的处理对象是集合而不是记录 各队,全队点名 --各队,全体点名! CREATE TABLE Teams (member CHAR(12) NOT NULL PRIMARY K ...

  9. SQL进阶系列之11让SQL飞起来

    写在前面 SQL的性能优化是数据库使用者必须面对的重要问题,本节侧重SQL写法上的优化,SQL的性能同时还受到具体数据库的功能特点影响,这些不在本节讨论范围之内 使用高效的查询 参数是子查询时,使用E ...

随机推荐

  1. java基础(三)之面向对象编程

    对象的创建方法 语法: class 类名{ 属性; 方法; } 生成对象的方法 类名 对象名 = new 类名(); Dog dog = new Dog(); 对象的使用方法1.对象.变量;2.对象. ...

  2. 实用 SQL 语句

    1. 创建 1.1 创建数据库 语法:create database db_name 示例:创建应用数据库 awesome_app sqlcreate database `awesome_app` 复 ...

  3. hdu2328 后缀树

    #include <cstdio> #include <cstdlib> #include <cstring> #include <cmath> #in ...

  4. jquery使用ajax实现实时刷新,轮询

    来自:https://blog.csdn.net/qq_25406669/article/details/78343082 var isLoaded = false; function reqs() ...

  5. 洛谷P1118 [USACO06FEB]数字三角形`Backward Digit Su`…

    #include<iostream> using namespace std ; ; int y[N][N]; int n; int a[N]; bool st[N]; int sum; ...

  6. spring boot 实战笔记(一)

    spring 概述: Bean :每一个被 Spring 管理的 JAVA对象,都称之为 Bean.Spring提供一个IoC容器来初始化对象,负责创建Bean, 解决对象之间的依赖管理和对象的使用. ...

  7. windows 安装 cordova

    windows 安装 cordova 参考资料:https://www.cnblogs.com/bpdxqx/p/6061719.html 1.安装nodejs(自动包含npm) 2.在命令行中通过n ...

  8. HTML5学习(6)a元素

    a元素代表超链接 href属性 hyper reference:通常代表跳转地址 target属性:_self在本窗口中打开(默认),_blank在新窗口中打开. id属性:全局属性,表示元素在文档中 ...

  9. command failed: npm install --loglevel error --registry=https://registry.npm 用vue-cli 4.0 新建项目总是报错

    昨天新买的本本,今天布环境,一安装vue-cli发现都4.0+的版本了,没管太多,就开始新建个项目感受哈,一切运行顺利,输入 "vue create app" 的时候,一切貌似进展 ...

  10. Oracle创表操作记录

    Oracle表操作 --主键,复合主键 create table example (id number primary key, name varchar2(20)); create table ex ...