百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 优雅编程 > 正文

常用SQL系列之(九):日期计算、分页、跳行与分级等查询

sinye56 2024-10-04 10:44 2 浏览 0 评论

本系列为@牛旦教育IT课堂在微头条上发布的内容,

为便于查阅,特辑录于此,都是常用SQL基本用法。

前两篇连接:

(一):SQL点滴(查询篇):数据库基础查询案例实战

(二):SQL点滴(排序篇):数据常规排序查询实战示例

(三):常用SQL系列之:记录叠加、匹配、外连接及笛卡尔等

(四):常用SQL系列之:Null值、插入方式、默认值及复制等

(五):常用SQL系列之:多表和禁止插入、批量与特殊更新等

(六):常用SQL系列之:删除方式、数据库、表及索引元信息查询等

(七):常用SQL系列之:表约束、最大/小值、非null数、平均值等

(八):常用SQL系列之:列值累计、占比、平均值以及日期运算等



SQL点滴(51):如何计算两个日期之间相差的月数或年数?


也就是确定两个日期间的月份数或年份数,计算其间差。比如计算第一个员工和最后一个员工聘用日期相差的月份数,以及这些月折合的年数。首先来看——
1)MySQL的实现示例参考:
SELECT
mnth,mnth / 12
FROM
(
SELECT
(YEAR( max_hd ) - YEAR ( min_hd ))*12 +
MONTH ( max_hd ) - MONTH ( min_hd ) AS mnth
FROM
( SELECT min( hiredate ) AS min_hd, max( hiredate ) AS max_hd FROM employee ) x
) y
注意这个写法,使用了函数Year和Month为给定的日期返回4位数年份和两位数月份。这个写法,也适用DB2。
2)MS SQL中的计算参考:
select datediff(month,min_hd,max_hd),datediff(month,min_hd,max_hd)/12
from( SELECT min( hiredate ) AS min_hd, max( hiredate ) AS max_hd FROM employee )x
若是Oracle中,其写法与此类似,只是所有内部函数不同,如下所示:
select months_between(min_hd,max_hd),months_between(min_hd,max_hd)/12
from( SELECT min( hiredate ) AS min_hd, max( hiredate ) AS max_hd FROM employee )x
这种计算,主要要想清楚他们的关系机理,你可以分别显示年份和月份以作对比,比如:
SELECT
YEAR( max_hd ), YEAR( min_hd ),
MONTH ( max_hd ),MONTH ( min_hd )
FROM
( SELECT min( hiredate ) AS min_hd, max( hiredate ) AS max_hd FROM employee ) x


好了,动手试试吧。

SQL点滴(52):如何实现对查询结果集的分页操作?


进一步讲,就是给查询结果分页,或者“滚动显示”所查询的结果。比如针对人员信息表有500条记录,希望每次显示10行,那么酱油50页可以“翻阅”,从第1页到第50页,在操作页面上,体现为1-50页,可以一页页“点击翻页”,顺序查看每一页,如何在SQL中实现呢?首先,我们来看:
1)在MySQL中的实现:
select id,col01,col02,col03,coln from sometable order by id limit 10 offset 0
上面的意思为我们以id为主键排序 ,第一次从第一条记录开始,一次查看前10条记录。需要注意的是,关键字limit 后的数值,指定要查看多少记录,offerset定义了返回结果的第一条记录的偏移位置(就是从排序集中哪条记录开始依次返回10条记录,或跳过哪些记录),如果体现在页面上,那么offer 0可以看做从头显示第一页10条记录,那么第二页的偏移量offset就是从(2-1)*10,以此类推。当然,具体返回那些字段,根据需要指定。
这种SQL的写法,也适合PostgreSQL。
2)在Oracle中的实现:
select col1,col2,coln
from(
select row_number() over (order by id) as rn,col1,col2,coln from sometable
) x where rn between 1 and 10 .
在where子句的between 和and后的数字,就是表示要返回从那条记录到哪条记录的返回值返回。
此写法同样适用DB2和MS SQL Server。row_number()给每行记录非配唯一序号,以便可以明确返回需要的页记录。
另外,对于Oracle,还有另外一种 方式,使用rownum来生成唯一记录号。
Oracle的ROWNUM的具体SQL实现,自己动手实现一下吧。

SQL点滴(53):小花招—怎样使用SQL语句来跳过表中的指定n行返回所有结果?


比如说,从employee表中每隔一行返回一名员工,或者查找第1、3、5等以此类推的记录,如何实现?其实这个实现可以基于我前面介绍的翻页模式来实现。具体来参考如下实现:
1)MySQL间隔模式返回结果:
SELECT
x.ename
FROM
(
SELECT
a.ename,
( SELECT count( * ) FROM employee b WHERE b.ename <= a.ename ) AS rn
FROM
employee a
) x
WHERE
MOD ( x.rn, 2 ) =1
由于MySQL没有为行分等级或给行编号的函数,所以需要用子查询给行分顺序等级(作为行编号,本查询中用名字来进行分级实现记录的行编号),然后使用求模操作来跳过行。
上面的SQL语句也适用PostgreSQL。
2)在Oracle中的实现:
select ename from (
select row_number() over (order by ename) rn,ename from employee
) x
where mod(rn,2) = 1
关于求模,上面的写法也适用DB2,若在MS SQL中,则用%运算符。上面的语句将跳过偶数行号。
自己动手试试吧。

SQL点滴(54):如何查询指定部门的员工和部门名称信息,而其余部门只显示部门名称而无员工信息?

比如说,只返回部门编号为2和3的部门员工信息以及部门信息,而编号为1和4的只返回部门信息,如何实现。其实,这个用外连接并在连接中用or逻辑即可。参考SQL如下:


SELECT
e.ename,
d.dpid,
d.dpname,
d.dpaddress
FROM
department d
LEFT JOIN employee e ON ( d.dpid = e.departid AND ( e.departid = 2 OR e.departid = 3 ) )
ORDER BY 1


注意,order by 1,是指按ename进行默认升序排序,如果是2,则指定位dpid升序排序。
这个写法适合MySQL、DB2、PostgreSQL以及SQL Server。也适用Oracle9i及以上版本。
另外,上面的语句还可以用e.departid进行筛选,然后进行外部链接,如下所示:
SELECT
e.ename,
d.dpid,
d.dpname,
d.dpaddress
FROM
department d
LEFT JOIN
( select ename,departid from employee where departid in(2,3)
) e on (e.departid = d.dpid )
order by 1
自己动手试试吧。如果有错,记得找找原因哦 ^_^

SQL点滴(55):如何通过查询语句为返回结果分级并返回感兴趣的结果?


比如说,对employee表的工资进行分级,然后返回想要的等级的相关数据。比如薪资一样的为同级别,不同的为不同级别。如果有10条记录,那分出来级别是小于等于10的。
在MySQL中,我使用了子查询,为每个工资创建了一个等级,然后利用等级做条件返回结果,参考语句如下:
第一步,可以查分级结果;
SELECT
( SELECT count( DISTINCT b.salary ) FROM employee b WHERE a.salary <= b.salary ) AS rnk,
a.salary,
a.ename
FROM
employee a order by 1
第二步,把分级结果作为查询对象,添加where子句来返回感兴趣的结果:
select * from x where rnk<=3
x指代第一步的查询语句(作为第二步的子查询),试试吧。我这里演示的两步结果如图所示。



另外,可以看看本号@牛旦教育IT课堂 中关于JPA学习文章:系列总结:JPA核心接口综合实战案例

相关推荐

linux 查看当前应用内存状况,以及内存参数含义

1、查看进程号ps-ef|greptomcat2、查看当前内存分配,200ms打印一次jstat-gc进程号2001jstat-gc344802001S0CS1C...

如何显示 Linux 系统上的可用内存?这几个命令很好用!

在Linux系统中,了解可用内存是优化系统性能、故障排查以及资源管理的重要一环。本文将详细介绍如何在Linux系统上显示可用内存,包括多种方法和工具的使用。在讨论可用内存之前,我们需要了解一些...

Linux 下查看内存使用情况方法总结

Q:我想监视Linux系统的内存使用情况,在Linux下有哪些视图或者命令行工具可用呢?在做Linux系统优化的时候,物理内存是其中最重要的一方面。自然的,Linux也提供了非常多的方法来监控宝贵的内...

2、linux命令-用户管理

linux命令-用户管理用户切换[root@eric~]#sueric#切换到用户eric[eric@ericroot]$[eric@ericroot]$su#切换到rootPas...

Centos 7 进入单用户模式详解

1、开机在启动菜单按e进入编辑模式找到linux16行,在最后添加init=/bin/sh编辑完后,按ctrl+x退出2、进单用户模式后,使用passwd修改密码,提示以下错误:passwd:Aut...

每日一个Linux命令解析——newusers

newusers:在Linux系统中,newusers是一个用于批量创建用户的命令。它从一个文件中读取多行用户信息,每行描述一个用户的详细信息,并根据这些信息创建多个用户或对现有用户进行批量修改。一...

openEuler操作系统管理员指南:管理用户与用户组

在Linux中,每个普通用户都有一个账户,包括用户名、密码和主目录等信息。除此之外,还有一些系统本身创建的特殊用户,它们具有特殊的意义,其中最重要的是管理员账户,默认用户名是root。同时Linux也...

Linux用户管理

1、用户信息文件/etc/passwdroot:x:0:0:root:/root:/bin/bash第一列:用户名第二列:密码位第三列:用户ID0超级用户UID。如果用户UID...

centos7基础-用户、组、权限管理

用户和组(1)用户、组、家目录的概念linux系统支持多用户,除了管理员,其他用户一般不应该使用root,而是应该向管理员申请一个账号。组类似于角色,系统可以通过组对有共性的用户进行统一管理。每个用户...

LINUX基础 ----------组及用户的概念

在Linux中,用户和组都是非常重要的概念,可以控制文件访问权限和资源的管理。用户是标识一个进程、应用程序或系统管理员的账号,Linux中每个用户用一个用户ID(UID)来标识。对于一个...

从零入门Linux(四)用户与权限管理

在Linux系统中,用户和权限管理是系统安全的重要组成部分。通过合理配置用户和权限,可以确保系统的安全性和资源的合理分配。以下是一些与用户和权限管理相关的常用命令和概念。1.用户管理1.1添加...

如何在 Linux 中管理用户?

在Linux系统中,用户是系统资源的主要使用者,每个用户都有一个唯一的标识符(用户ID)。为了更好地组织和管理用户,Linux还引入了用户组的概念。用户组是用户的集合,有助于更有效地分配权限和资...

在 Linux 中将用户添加到特定组的四种方法

在Linux多用户操作系统中,用户组管理是系统安全架构的基石。通过合理的组权限分配,管理员可以实现:精确控制文件访问权限(chmod775project/)简化批量用户权限管理(setfacl-...

我不是网管 - 如何在Ubuntu Linux下创建sudo用户

Sudo用户是Linux系统的普通用户,具有一定的管理权限,可以对系统执行管理任务。在Linux中,root是超级用户,拥有完全的管理权限,但不建议将root凭证授予其他用户或作为r...

Linux创建普通用户,为密钥方式登录做准备

Hi,我是聪慧苹果8,就是江湖上人见人爱、花见花开,土到掉榨的Linux爱好者,一起学习吧!上一篇关于SSH安全加固的文字,有网友点评通过密钥登录更加安全,先创建一个普通用户,拒绝直接使用密码登录,这...

取消回复欢迎 发表评论: