sql获取分组后取某字段最大一条记录





<!--2018-1-9 11:53 zxk 根据宿舍号码判断判断是否请假   / 2018-1-11 10:49 第二次修改,还需要添加一个条件楼号-->
   <select id="findListApplyZIf" parameterClass="HashMap" resultClass="com.dwz.entity.DormZ">
   	<![CDATA[
   	     SELECT
		*
	     FROM
		student st
		LEFT JOIN
			(				
			  SELECT 
			     a.xh AS axh,a.gotime AS gotime,a.backtime AS backtime
			  FROM
			     apply a
			     inner join (SELECT xh,max(gotime) AS gy FROM apply GROUP BY xh) y ON a.xh=y.xh AND a.gotime=y.gy					
                        ) ay
		ON st.xh = ay.axh
		LEFT JOIN roll_call rc ON st.id = rc.stuid 
	     WHERE 1=1 
         ]]>
   	<!--2018-1-11 10:49 第二次修改,还需要添加一个条件楼号-->
   	<isNotEmpty property="bulidingnumZ">
   		AND st.bulidingnum = #bulidingnumZ#
   	</isNotEmpty>
   	<isNotEmpty property="dormnum">
   		AND st.dormnum = #dormnum#
   	</isNotEmpty>
   	<isNotEmpty property="xm">
   		AND st.xm = #xm#
   	</isNotEmpty>
   	<isNotEmpty property="sid">
   		AND st.sid = #sid#
   	</isNotEmpty>
   </select>

applay表中:这里面可以有多个学生的申请,每个学生可以有多个申请,时间都不同,我需要得到这个学生中gotime最大的这条数据

1.

得到所有xh的最大gotime(去重)
 SELECT   
    xh,max(gotime) AS gy 
 FROM 
    apply 
 GROUP BY xh

 因为我们还需要得到a.backtime,所以要用inner join内连接得到
 SELECT 
    a.xh AS axh,a.gotime AS gotime,a.backtime AS backtime
 FROM
    apply a
    inner join (
                SELECT   
                   xh,max(gotime) AS gy 
                FROM 
                   apply 
                GROUP BY xh
               ) y 
    ON a.xh=y.xh AND a.gotime=y.gy	

2.

得到某个xh最大的时间gotime:
 SELECT 
    xh,max(gotime) AS gy 
 FROM 
    apply 

当需要得到所有xh的最大时间gotime时(去重):
 SELECT
    a.xh AS axh,a.gotime AS gotime,a.backtime AS backtime 
 FROM apply AS a 
 WHERE gotime = (
                  SELECT 
                     max(b.gotime) 
                  FROM apply AS b
                  WHERE a.xh = b.xh
                 )  



  
### SQL 分组每组一条记录 为了实现在SQL分组获取每组一条记录,可以采用多种方法。具体实现决于所使用的数据库管理系统以及需求的具体细节。 #### 方法一:使用窗口函数 现代版本的大多数关系型数据库支持窗口函数,这使得操作变得更为简便。通过`ROW_NUMBER()`这样的窗口函数,可以在每一组内部创建一个按某些标准排序后的行号,从而方便选各组的第一条记录[^3]。 ```sql WITH RankedRecords AS ( SELECT *, ROW_NUMBER() OVER (PARTITION BY team_id ORDER BY pageviews DESC) as rn FROM your_table_name ) SELECT id, name, team_id, pageviews FROM RankedRecords WHERE rn = 1; ``` 此代码片段展示了如何利用`ROW_NUMBER()`为每个团队分配基于页面浏览量降序排列的独特编号,并最终筛选出具有最小行号(`rn=1`)即最高页面浏览量的那一项。 #### 方法二:子查询方式 如果不支持窗口函数,则可以通过自连接或者嵌套子查询的方式来达到相同的效果。这种方法通常效率较低,但在不支持高级特性的旧版DBMS上仍然适用[^4]。 ```sql SELECT t1.* FROM your_table_name t1 LEFT JOIN your_table_name t2 ON t1.team_id = t2.team_id AND (t1.pageviews < t2.pageviews OR (t1.pageviews = t2.pageviews AND t1.id < t2.id)) WHERE t2.id IS NULL; ``` 上述语句逻辑在于找到那些在其所属小组内没有任何其他成员拥有更大或相等(但ID更小)pageview值的记录;换句话说,这些正是我们要找寻的目标——每组中的最大者之一。 #### 方法三:聚合函数配合外键关联 当只需要某类特定字段而不需要完整的原始行时,可以直接运用聚合函数如MAX(), MIN()等结合GROUP BY完成任务。不过需要注意的是这种方式无法保留原表结构之外的信息[^1]。 ```sql SELECT MAX(id), NAME, team_id, MAX(pageviews) FROM your_table_name GROUP BY team_id, NAME; -- 假设name在team_id下唯一 ``` 以上三种方案提供了不同场景下的解决方案,选择哪种决于实际应用场景和个人偏好。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值