当前位置: 首页 > 图文教程 > 数据库 > MSSQL > 一条语句简单解决“每个Y的最新X”的经典sql语句

MSSQL
SQLServer中求两个字符串的交集
SQL Server 触发器 表的特定字段更新时,触发Update触发器
SQL 查询分析中使用net命令问题
SQLServer 批量导入目录文件
sql语言中delete删除命令语句详解
Linq to SQL 插入数据时的一个问题
一个简单的SQL 行列转换语句
SQLServer 常用语句(按功能分)
SQLServer 管理常用SQL语句
SQL Server 2000中的触发器使用
SQL小技巧 又快又简单的得到你的数据库每个表的记录数
Sql Server 2000删除数据库备份文件
批量更新数据库所有表中字段的内容,中木马后的急救处理
sqlserver 通用分页存储过程
远程连接局域网内的sql server 无法连接 错误与解决方法
目前用到的两个分页存储过程代码
sqlserver 多表关联时在where语句中慎用trim()方法
Sql2005注射辅助脚本[粗糙版]
sql 2005不允许进行远程连接可能会导致此失败的解决方法
MSSQL2005数据库备份导入MSSQL2000

MSSQL 中的 一条语句简单解决“每个Y的最新X”的经典sql语句


出处:互联网   整理: 软晨网(RuanChen.com)   发布: 2009-09-13   浏览: 108 ::
收藏到网摘: n/a

“每个Y的最新X”是一个经典的SQL问题,工作中经常碰到。当然不是“按Y分组求最新的X值”那么简单,要求最新X的那条记录或主键ID。用一条SQL语句可以简单的解决此问题。 生成实例表和数据:
复制代码 代码如下:

/****** 创建表 ******/
if exists (select * from dbo.sysobjects where id = object_id(N'[dbo].[Table]') and OBJECTPROPERTY(id, N'IsUserTable') = 1)
drop table [dbo].[Table]
GO
Create TABLE [dbo].[Table] (
[ID] [int] IDENTITY (1, 1) NOT NULL ,
[Y] [nvarchar] (50) COLLATE Chinese_PRC_CI_AS NOT NULL ,
[X] [smalldatetime] NOT NULL
) ON [PRIMARY]
GO
--插入数据
Insert INTO [Table](Y, X) values('CCC', '2007-02-02 02:02:02')
Insert INTO [Table](Y, X) values('AAA', '2007-02-02 02:02:02')
Insert INTO [Table](Y, X) values('AAA', '2007-03-03 03:03:03')
Insert INTO [Table](Y, X) values('BBB', '2007-01-01 01:01:01')
Insert INTO [Table](Y, X) values('BBB', '2007-02-02 02:02:02')
Insert INTO [Table](Y, X) values('BBB', '2007-03-03 03:03:03')
Insert INTO [Table](Y, X) values('CCC', '2007-01-01 01:01:01')
Insert INTO [Table](Y, X) values('AAA', '2007-01-01 01:01:01')
Insert INTO [Table](Y, X) values('CCC', '2007-03-03 03:03:03')
Insert INTO [Table](Y, X) values('DDD', '2007-01-01 01:01:01')
Insert INTO [Table](Y, X) values('DDD', '2007-02-02 02:02:02')
Insert INTO [Table](Y, X) values('DDD', '2007-03-03 03:03:03')
Insert INTO [Table](Y, X) values('EEE', '2007-01-01 01:01:01')
Insert INTO [Table](Y, X) values('EEE', '2007-02-02 02:02:02')
Insert INTO [Table](Y, X) values('EEE', '2007-03-03 03:03:03')
GO
/****** 创建表 ******/
if exists (select * from dbo.sysobjects where id = object_id(N'[dbo].[Table]') and OBJECTPROPERTY(id, N'IsUserTable') = 1)
drop table [dbo].[Table]
GO
Create TABLE [dbo].[Table] (
[ID] [int] IDENTITY (1, 1) NOT NULL ,
[Y] [nvarchar] (50) COLLATE Chinese_PRC_CI_AS NOT NULL ,
[X] [smalldatetime] NOT NULL
) ON [PRIMARY]
GO
--插入数据
Insert INTO [Table](Y, X) values('CCC', '2007-02-02 02:02:02')
Insert INTO [Table](Y, X) values('AAA', '2007-02-02 02:02:02')
Insert INTO [Table](Y, X) values('AAA', '2007-03-03 03:03:03')
Insert INTO [Table](Y, X) values('BBB', '2007-01-01 01:01:01')
Insert INTO [Table](Y, X) values('BBB', '2007-02-02 02:02:02')
Insert INTO [Table](Y, X) values('BBB', '2007-03-03 03:03:03')
Insert INTO [Table](Y, X) values('CCC', '2007-01-01 01:01:01')
Insert INTO [Table](Y, X) values('AAA', '2007-01-01 01:01:01')
Insert INTO [Table](Y, X) values('CCC', '2007-03-03 03:03:03')
Insert INTO [Table](Y, X) values('DDD', '2007-01-01 01:01:01')
Insert INTO [Table](Y, X) values('DDD', '2007-02-02 02:02:02')
Insert INTO [Table](Y, X) values('DDD', '2007-03-03 03:03:03')
Insert INTO [Table](Y, X) values('EEE', '2007-01-01 01:01:01')
Insert INTO [Table](Y, X) values('EEE', '2007-02-02 02:02:02')
Insert INTO [Table](Y, X) values('EEE', '2007-03-03 03:03:03')

GO解决“每个Y的最新X”经典SQL问题:以下几种方法真是八仙过海
复制代码 代码如下:

Select ID, Y, X
FROM [Table] T1
Where (NOT EXISTS
(Select 1
FROM [Table] T2
Where (T2.Y = T1 .Y) AND (T2.X > T1 .X or
T2.X = T1 .X AND T2.ID > T1 .ID)))
/*****************************************************************************/
Select *
FROM [Table]
Where ID IN
(Select MAX(T1.ID)
FROM [Table] T1 JOIN
(Select y, MAX(x) x
FROM [Table]
GROUP BY y) T2 ON T1.Y = T2.Y AND T1.x = T2.x
GROUP BY T1.y)
/*****************************************************************************/
Select T .ID, T .Y, T .X
FROM [Table] T INNER JOIN
(Select MAX(T1.ID) AS ID
FROM [Table] T1 JOIN
(Select y, MAX(x) x
FROM [Table]
GROUP BY y) T2 ON T1.Y = T2.Y AND T1.x = T2.x
GROUP BY T1.y) T2 ON T .ID = T2.ID
/*****************************************************************************/
Select *
FROM [Table] T1
Where ID IN
(Select TOP 1 ID
FROM [Table]
Where Y = T1.Y
orDER BY X DESC)
/*****************************************************************************/
Select *
FROM [Table] T1
Where (ID =
(Select TOP 1 ID
FROM [Table]
Where Y = T1.Y
orDER BY X DESC, ID DESC))
/*****************************************************************************/
/*****************************************************************************/
Select ID, Y, X
FROM [Table] T1
Where (NOT EXISTS
(Select 1
FROM [Table] T2
Where (T2.Y = T1 .Y) AND (T2.X > T1 .X or
T2.X = T1 .X AND T2.ID > T1 .ID)))
/*****************************************************************************/
Select *
FROM [Table]
Where ID IN
(Select MAX(T1.ID)
FROM [Table] T1 JOIN
(Select y, MAX(x) x
FROM [Table]
GROUP BY y) T2 ON T1.Y = T2.Y AND T1.x = T2.x
GROUP BY T1.y)
/*****************************************************************************/
Select T .ID, T .Y, T .X
FROM [Table] T INNER JOIN
(Select MAX(T1.ID) AS ID
FROM [Table] T1 JOIN
(Select y, MAX(x) x
FROM [Table]
GROUP BY y) T2 ON T1.Y = T2.Y AND T1.x = T2.x
GROUP BY T1.y) T2 ON T .ID = T2.ID
/*****************************************************************************/
Select *
FROM [Table] T1
Where ID IN
(Select TOP 1 ID
FROM [Table]
Where Y = T1.Y
orDER BY X DESC)
/*****************************************************************************/
Select *
FROM [Table] T1
Where (ID =
(Select TOP 1 ID
FROM [Table]
Where Y = T1.Y
orDER BY X DESC, ID DESC))

/*****************************************************************************
/效率嘛,在不同的字段建立索引速度都不尽相同,使用者见仁见智了.
第一种方法速度在各方面都不错,而且在Y列在建立索引,可以大大优化查询速度。