被忽视但很实用的那部分SQL

一、前言

虽然我们大多数人都学习过SQL,但是经常忽略它。总是会自以为学到的已经足够用了,从而导致我们在实际开发的过程中遇到复杂的问题后只能在检索数据后通过传统的代码来完成,但是其中很多的功能利用SQL就可以轻松的办到,所以我们开始重视SQL,它的地位不亚于C#javascript

二、目录

1.多行插入

2.将其他表数据插入

3.子查询

4.通用表表达式(CTE)

5.MERGE指令

6.窗口化函数

7.查询分页

三、正文

1.多行插入

大家都学习过INSERT,但是每次我们都只插入一条数据,如果我们需要插入多条数据呢,那么情况就会和图1.1一样。

图1.1

上面是最简单的示例,我们还可以进行一定的优化,避免循环建立和关闭连接的次数,比如下图(图1.2

图1.2

即使我们采用了图1.2所示的方式,既然会长时间的占据带框,并且Web服务器和数据库服务器既然要进行多次连接,如果使用本节的知识我们就可以做到图1.3所示的效果。

图1.3

首先我们先来看看普通的插入语句:

1 INSERT INTO Cou_Course(CourseID,TeacherID,SemesterID,CourseName,CourseType,CourseState)
2 VALUES(‘235648651235423512‘,‘12312421421‘,‘12124214124‘,‘awefweagwaeg‘,‘123‘,‘124124‘)

每次插入一条新的数据都要重新一次,如果是批量的添加就可想而知了,而如果我们直接通过一条插入语句插入多条数据就可以这样做:

1 INSERT INTO Cou_Course(CourseID,TeacherID,SemesterID,CourseName,CourseType,CourseState)
2 VALUES(‘235648651235423515‘,‘12312421421‘,‘12124214124‘,‘awefweagwaeg‘,‘123‘,‘124124‘),
3 (‘235648651235423514‘,‘12312421421‘,‘12124214124‘,‘awefweagwaeg‘,‘123‘,‘124124‘),
4 (‘235648651235423517‘,‘12312421421‘,‘12124214124‘,‘awefweagwaeg‘,‘123‘,‘124124‘)

当然语法也很简单,只要用逗号来分割就可以了。虽然仅仅只是一个简单的技巧,但是又有多少人会使用呢?

2.将其他表数据插入

解决了批量插入的问题,我们还会遇到将其他表的数据插入到另一个表中的需求,特别是在编写存储过程中我们可能会对数据进行处理,并把处理的结果存放在临时 表中,最后才将临时表中的数据全部添加到实际的表中,如果利用传统的做法会非常麻烦,而通过本节介绍的知识将可以很轻松的实现。

首先我们需要做一些简单的准备工作,这里我们需要创建两个表,A表(id为自增),B表跟A表一样,然后我们在A表中随意写一些内容:

然后利用 INSERT INTOSELECT … 来实现将A表的数据添加到B表中去:

INSERT INTO B(Name) SELECT Name FROM A

当然读者看过之后会觉得非常简单,因为我们现在仅仅只是学习这个知识点,重点在于我们能够记住它,并能够在后面使用它(实际开发中会比这更复杂,但是基本的东西还是不变的)

3.子查询

在正常的使用中我们经常会遇到无法使用链接的情况,但是我们需要根据另一个表的情况来决定当前的SELECT,这个时候我们就需要子查询。笔者在这里只能例举简单的示例,实际的需求会出现非常复杂的子查询。这里我们还是借助上面的A、B表,首先实现的功能是检索A表,但是条件是A表的Name要在B表中存在:

SELECT id,Name FROM A
WHERE EXISTS(
    SELECT Name FROM B
    WHERE B.Name = A.Name
)

上面的语句中我们在WHERE中就使用了子查询负责去B表中查询是否存在记录,而EXISTS的作用就是EXISTS括号中的语句只要返回了一个或一个以上的结果则成立。所以最后我们可以看到所有的数据都呈现了。当然子查询不仅仅可以用于条件中,我们还可以用于SELECT后,比如我们根据A表的Name去B表中查询Name跟它一样的id号:

SELECT (SELECT TOP 1 B.id FROM B WHERE B.Name = A.Name) FROM A

子查询的用处非常多,也非常强大,远不止笔者这里介绍的这么一点。

4.通用表表达式(CTE)

虽然这个名字很专业,但是实际用起来是非常简单的,当然简单的同时也帮助我们解决了很多的问题,最终的效果跟临时表一样,但是使用起来比临时表更方便。比如下面的代码我们将创建一个名为MyCTE的临时表:

WITH MyCTE AS(
  SELECT * FROM A
)
SELECT * FROM MyCTE

是不是非常简单,当然还可以同时定义多个通用表,比如下面的代码所示:

1 WITH MyCTE AS(
2   SELECT * FROM A
3 ), MyCTE2 AS(
4   SELECT * FROM B
5 )
6 SELECT * FROM MyCTE,MyCTE2

多个通用表只需要用逗号分割即可,当然我们这里还涉及了一个知识点,相信有人会发觉出来。

5.MERGE指令

这个指令是SQL SERVER 2008中 新增的,相比前面几个来说比较难懂,但是作用却非常强大,利用这个指令我们可以同时进行添加、修改和删除,并且是由条件的。具体的实现方式就是根据源表与 目标表进行对比,如果匹配则执行对应的更新操作,如果源表中存在,但是目标表不存在则执行添加操作,如果相反则执行删除操作。下面我们将通过循序渐进的方 式来介绍如何使用MERGE,首先我们需要确定目标表,因为后面的更新,添加和删除操作都是针对目标表的,所以目标表只能是一个表不能是检索后的数据,比如下面的代码我们将前面我们示例中使用的A表作为目标表:

1 MERGE A AS itarget

有了目标表还不足够,我们还要需要一个源表,用来形成对比,而源表则可以是检索语句,因为笔者这里只是简单的示例,所以直接检索了B表中的数据:

1 USING(
2   SELECT * FROM B
3 ) AS isource

这样我们就有了目标表和源表,最后合并:

1 MERGE A AS itarget
2 USING(
3   SELECT * FROM B
4 ) AS isource

接着我们需要指定对应的条件,从而根据是否符合这个条件而决定对目标表进行什么操作,比如下面的语句将判断两表中是否存在相同id的数据:

ON (itarget.id = isource.id)

有了条件后,我们就可以根据这个条件进行对应的操作了,笔者将在满足这个条件后修改目标表的Name,在后面追加change字符串:

1 WHEN MATCHED THEN
2 UPDATE SET itarget.Name = itarget.Name + ‘change‘

最后的语句如下所示:

1 MERGE A AS itarget
2 USING(
3   SELECT * FROM B
4 ) AS isource
5 ON (itarget.id = isource.id)
6 WHEN MATCHED THEN
7 UPDATE SET itarget.Name = itarget.Name + ‘change‘;

最后我们查看A表,发现数据都改变了:

这个时候我们在A表新添加一条数据,以满足源表不匹配的情况,然后在原本的语句后面添加如下的语句:

WHEN NOT MATCHED BY SOURCE THEN
DELETE;

我们可以猜测出,当目标表中存在源表中不存在的数据后将会删除这条数据,所以执行后我们将看到A表新添加的数据已经被删除了,完整的语句如下所示:

1 MERGE A AS itarget
2 USING(
3   SELECT * FROM B
4 ) AS isource
5 ON (itarget.id = isource.id)
6 WHEN MATCHED THEN
7 UPDATE SET itarget.Name = itarget.Name
8 WHEN NOT MATCHED BY SOURCE THEN
9 DELETE;

最后就是源表中存在,但是目标表中不存在的情况了,我们只需要将上面的BY SOURCE改成BY TARGET即可,通过下面这条语句我们将把源表与目标表不匹配的数据添加到目标表中去(当然我们需要提前在源表中新增一条数据)

WHEN NOT MATCHED BY TARGET THEN
INSERT (Name) VALUES(isource.Name);

执行完成后我们将看到A表多了几条数据。下面是完整的语句:

 1 MERGE A AS itarget
 2 USING(
 3   SELECT * FROM B
 4 ) AS isource
 5 ON (itarget.id = isource.id)
 6 WHEN MATCHED THEN
 7 UPDATE SET itarget.Name = itarget.Name
 8 WHEN NOT MATCHED BY SOURCE THEN
 9 DELETE
10 WHEN NOT MATCHED BY TARGET THEN
11 INSERT (Name) VALUES(isource.Name);

6.窗口化函数

首先是ROW_NUMBER,顾名思义,就是给我们检索出来的数据加上序号,旧的分页都是采用这种方式,但是往往我们仅仅只是使用了它的一点,他还可以分块进行标序号,比如我们将上面的A表改成如下形式:

然后采用如下所示的SQL语句,就可以按照Name进行标序号:

SELECT ROW_NUMBER() OVER(PARTITION BY A.Name ORDER BY A.id) AS ‘RNUM‘,id FROM A

结果如下所示:

下面我们通过一段SQL以及对应的结果呈现其他的窗口化函数:

SELECT ROW_NUMBER() OVER(PARTITION BY A.Name ORDER BY A.id) AS ‘RNUM‘,
RANK() OVER(ORDER BY A.Name) AS ‘RANK‘,
DENSE_RANK() OVER(ORDER BY A.Name) AS ‘DENSE RANK‘,
NTILE(4) OVER(ORDER BY A.id) AS ‘NTILE‘ FROM A

结果如下所示:

其中简单介绍下NTILE,我们传了一个4那么它会将前面1/4标记为1,然后接着标记1/4为2,以此类推。关于RANKDENSE_RANK比较好理解,看看最后的结果就可以得出结论了。

7.分页查询

这是最后一节了,但是相关的语句却很简单,我们只要记住以下关键字就可以了:

OFFSET…FETCH  NEXT…

比如下面的SQL语句,我们将跳过前面5条数据,获取3条数据:

1 SELECT * FROM A
2 ORDER BY A.id
3 OFFSET 5 ROWS
4 FETCH NEXT 3 ROWS ONLY
时间: 2024-10-24 05:45:42

被忽视但很实用的那部分SQL的相关文章

一个很实用的前端框架Zui

杰哥给我推荐了一个很有用的前端框架-Zui,我看着觉得很神奇的,因为有很多我都不懂.在这里分享总结一下.首先,这是一个中国自己开发的框架,比起很多外国的框架来说,有很详细的API,而且是全中文的,不需要再经过其他人的翻译了.然后,它的内容十分丰富,很系统的分为了:基础,控件,组件,JS插件,视图几大块:而且使用起来,只需要导入js,在适当的地方加上正确的class类就可以了.对于,没有什么js基础的人,也是十分容易上手的.下面我就大体的介绍一下它的各个模块的功能.基础:基础里面我觉得很有用的主要

分享15款很实用的 Sass 和 Compass 工具

Sass 是 CSS 的扩展,增加了嵌套规则,变量,混入功能等很多更多.它简化了组织和维护 CSS 代码的成本.Compass 是一个开源的 CSS 框架,使得使用 CSS3 和流行的设计模式比以往任何时候都更容易. 在这篇文章中,我们已经收集了一组有用的 Sass 和 Compass 工具,将帮助您快速构建 Web 应用程序. 您可能感兴趣的相关文章 网站开发中很有用的 jQuery 效果[附源码] 分享35个让人惊讶的 CSS3 动画效果演示 十分惊艳的8个 HTML5 & JavaScri

Web 开发中很实用的10个效果【附源码下载】

在工作中,我们可能会用到各种交互效果.而这些效果在平常翻看文章的时候碰到很多,但是一时半会又想不起来在哪,所以养成知识整理的习惯是很有必要的.这篇文章给大家推荐10个在 Web 开发中很有用的效果,记得收藏:) 超炫的页面切换动画效果 今天我们想与大家分享一组创意的页面切换熊效果集合.我们已经在示例中罗列了一组动画,可以被应用到页面切换过程中,创造出很有趣的导航效果. 立即下载      在线演示 美!视差滚动在图片滑块中的应用 视差滚动(Parallax Scrolling)已经被广泛应用于网

java 获取公网(外网IP)很实用!

package com.lovo.util; import java.io.BufferedReader; import java.io.InputStream; import java.io.InputStreamReader; import java.net.HttpURLConnection; import java.net.URL; public class PublicInterIp { /** * @param args * @throws Exception */ public S

修改html很实用的insertAdjacentHTML方法

添加HTML内容与文本内容以前用的是innerHTML与innerText方法, 最近发现还有insertAdjacentHTML和 insertAdjacentText方法, 这两个方法更灵活,可以在指定的地方插入html内容和文本内容. insertAdjacentText方法与 insertAdjacentHTML方法类似,只不过只能插入纯文本,参数相同 原型:insertAdajcentHTML(swhere,stext) insertAdjacentHTML方法:在指定的地方插入htm

CSS3中的伪类概览,很实用的

CSS的目标,或者说是革命纲领就是:"将样式与内容分离",这个目标能否实现,很大程度上依赖于CSS访问内容的能力.在CSS3中,这个任务得到了有力支持,强大的结构伪类Structural pseudo-classes出现了.提供了非常丰富的查询方式,让CSS选择器变得如此和蔼可亲了.让我们来快速概览一下,同时期待着浏览器整体换代的来到,毕竟只有最新的浏览器才支持CSS3选择器及其他特性. 支持浏览器:Firefox 3.1+, IE8+ (only in IE8 standards m

简单但很实用的js手风琴效果

<!DOCTYPE html> <html> <head>     <meta charset="UTF-8">     <title></title>     <style> body,ul,li,.sfq,h3{     margin: 0;     padding: 0;     list-style: none; }   h3{       height: 30px;       width: 

【转载】Web 开发中很实用的10个效果【附源码下载】

超炫的页面切换动画效果 今天我们想与大家分享一组创意的页面切换熊效果集合.我们已经在示例中罗列了一组动画,可以被应用到页面切换过程中,创造出很有趣的导航效果. 立即下载      在线演示 美!视差滚动在图片滑块中的应用 视差滚动(Parallax Scrolling)已经被广泛应用于网页设计中,这种技术能够让原本平面的网页界面产生动感的立体效果.美女很养眼吧 :) 源码下载      在线演示 网页边栏过渡动画 以细微的过渡动画显示一些隐藏的侧边栏,其余的内容也是.通常侧边栏滑入,把其他内容推

IOS如果将一个十六进制的color转换成UIColor,很实用

UI给开发的效果图很多时候标注着十六进制的Color,而程序中用到的往往是UIColor可以用如下方法去转换: (UIColor *)RGBColorFromHexString:(NSString *)color alpha:(float)alpha { //color的值类似#fffeee,alpha的值类似1.0为透明度 int nums[6] = {0}; for (int i = 1; i < [color length]; i++) { int temp = [color charac