`

SQL SERVER性能优化综述(很好的总结,不要错过哦)第2/3页

阅读更多

F、
关于临时表产生使用SELECT INTO和CREATE TABLE + INSERT INTO的选择,我们做过测试,一般情况下,SELECT INTO会比CREATE TABLE + INSERT INTO的方法快很多,但是SELECT INTO会锁定TEMPDB的系统表SYSOBJECTS、SYSINDEXES、SYSCOLUMNS,在多用户并发环境下,容易阻塞其他进程,所以我的建议是,在并发系统中,尽量使用CREATE TABLE + INSERT INTO,而大数据量的单个语句使用中,使用SELECT INTO。
G、
注意排序规则,用CREATE TABLE建立的临时表,如果不指定字段的排序规则,会选择TEMPDB的默认排序规则,而不是当前数据库的排序规则。如果当前数据库的排序规则和TEMPDB的排序规则不同,连接的时候就会出现排序规则的冲突错误。一般可以在CREATE TABLE建立临时表时指定字段的排序规则为DATABASE_DEFAULT来避免上述问题。
5、
子查询的用法
子查询是一个 SELECT 查询,它嵌套在 SELECT、INSERT、UPDATE、DELETE 语句或其它子查询中。任何允许使用表达式的地方都可以使用子查询。
子查询可以使我们的编程灵活多样,可以用来实现一些特殊的功能。但是在性能上,往往一个不合适的子查询用法会形成一个性能瓶颈。
如果子查询的条件中使用了其外层的表的字段,这种子查询就叫作相关子查询。相关子查询可以用IN、NOT IN、EXISTS、NOT EXISTS引入。
关于相关子查询,应该注意:
A、NOT IN、NOT EXISTS的相关子查询可以改用LEFT JOIN代替写法。比如:
SELECT PUB_NAMEFROM PUBLISHERS
WHERE PUB_ID NOT IN (SELECT PUB_ID
FROM TITLES
WHERE TYPE = 'BUSINESS')
可以改写成:
SELECT A.PUB_NAMEFROM PUBLISHERS A LEFT JOIN TITLES B
ON B.TYPE = 'BUSINESS' AND A.PUB_ID=B. PUB_ID
WHERE B.PUB_ID IS NULL
SELECT TITLEFROM TITLES
WHERE NOT EXISTS (SELECT TITLE_ID
FROM SALES
WHERE TITLE_ID = TITLES.TITLE_ID)
可以改写成:
SELECT TITLEFROM TITLES LEFT JOIN SALES
ON SALES.TITLE_ID = TITLES.TITLE_ID
WHERE SALES.TITLE_ID IS NULL
B、
如果保证子查询没有重复 ,IN、EXISTS的相关子查询可以用INNER JOIN 代替。比如:
SELECT PUB_NAMEFROM PUBLISHERS
WHERE PUB_ID IN (SELECT PUB_ID
FROM TITLES
WHERE TYPE = 'BUSINESS')
可以改写成:
SELECT DISTINCT A.PUB_NAMEFROM PUBLISHERS A INNER JOIN TITLES B
ON B.TYPE = 'BUSINESS' AND
A.PUB_ID=B. PUB_ID
C、
IN的相关子查询用EXISTS代替,比如
SELECT PUB_NAMEFROM PUBLISHERS
WHERE PUB_ID IN (SELECT PUB_ID
FROM TITLES
WHERE TYPE = 'BUSINESS')
可以用下面语句代替:
SELECT PUB_NAMEFROM PUBLISHERS
WHERE EXISTS (SELECT 1
FROM TITLES WHERE TYPE = 'BUSINESS' AND
PUB_ID= PUBLISHERS.PUB_ID)
D、不要用COUNT(*)的子查询判断是否存在记录,最好用LEFT JOIN或者EXISTS,比如有人写这样的语句:
SELECT JOB_DESC FROM JOBS
WHERE (SELECT COUNT(*) FROM EMPLOYEE WHERE JOB_ID=JOBS.JOB_ID)=0
应该改成:
SELECT JOBS.JOB_DESC FROM JOBS LEFT JOIN EMPLOYEE ON EMPLOYEE.JOB_ID=JOBS.JOB_ID
WHERE EMPLOYEE.EMP_ID IS NULL
SELECT JOB_DESC FROM JOBS
WHERE (SELECT COUNT(*) FROM EMPLOYEE WHERE JOB_ID=JOBS.JOB_ID)<>0
应该改成:
SELECT JOB_DESC FROM JOBS
WHERE EXISTS (SELECT 1 FROM EMPLOYEE WHERE JOB_ID=JOBS.JOB_ID)
6、
慎用游标
数据库一般的操作是集合操作,也就是对由WHERE子句和选择列确定的结果集作集合操作,游标是提供的一个非集合操作的途径。一般情况下,游标实现的功能往往相当于客户端的一个循环实现的功能,所以,大部分情况下,我们把游标功能搬到客户端。
游标是把结果集放在服务器内存,并通过循环一条一条处理记录,对数据库资源(特别是内存和锁资源)的消耗是非常大的,所以,我们应该只有在没有其他方法的情况下才使用游标。
另外,我们可以用SQL SERVER的一些特性来代替游标,达到提高速度的目的。
A、字符串连接的例子
这是论坛经常有的例子,就是把一个表符合条件的记录的某个字符串字段连接成一个变量。比如需要把JOB_ID=10的EMPLOYEE的FNAME连接在一起,用逗号连接,可能最容易想到的是用游标:
DECLARE @[url=URL]NAME[/url] VARCHAR(20) DECLARE @NAME VARCHAR(1000)
DECLARE NAME_CURSOR CURSOR FOR SELECT FNAME FROM EMPLOYEE WHERE JOB_ID=10 ORDER BY EMP_ID
OPEN NAME_CURSOR FETCH NEXT FROM RNAME_CURSOR INTO @NAME
WHILE @@FETCH_STATUS = 0
BEGIN
SET @NAMES = ISNULL(@NAMES+',','')+@NAME FETCH NEXT FROM NAME_CURSOR INTO @NAME
END CLOSE NAME_CURSOR
DEALLOCATE NAME_CURSOR
可以如下修改,功能相同:
DECLARE @NAME VARCHAR(1000) SELECT @NAMES = ISNULL(@NAMES+',','')+FNAME
FROM EMPLOYEE WHERE JOB_ID=10 ORDER BY EMP_ID
B、
用CASE WHEN 实现转换的例子
很多使用游标的原因是因为有些处理需要根据记录的各种情况需要作不同的处理,实际上这种情况,我们可以用CASE WHEN语句进行必要的判断处理,而且CASE WHEN是可以嵌套的。比如:
表结构:
CREATE TABLE 料件表(料号 VARCHAR(30),
名称 VARCHAR(100),主单位 VARCHAR(20),
单位1 VARCHAR(20),单位1参数 NUMERIC(18,4),
单位2 VARCHAR(20),单位2参数 NUMERIC(18,4)
)
GO
CREATE TABLE 入库表(时间 DATETIME,
料号 VARCHAR(30),单位 INT,
入库数量 NUMERIC(18,4),损坏数量 NUMERIC(18,4)
)
GO
其中,单位字段可以是0,1,2,分别代表主单位、单位1、单位2,很多计算需要统一单位,统一单位可以用游标实现:
DECLARE @料号 VARCHAR(30), @单位 INT,
@参数 NUMERIC(18,4),
DECLARE CUR CURSOR FOR SELECT 料号,单位 FROM 入库表 WHERE 单位 <>0
OPEN CURFETCH NEXT FROM CUR INTO @料号,@单位
WHILE @@FETCH_STATUS<>-1
BEGIN
IF @单位=1
BEGIN
SET @参数=(SELECT 单位1参数 FROM 料件表 WHERE 料号 =@料号) UPDATE 入库表 SET 数量=数量*@参数,损坏数量=损坏数量*@参数,单位=1 WHERE CURRENT OF CUR
END IF @单位=2
BEGIN SET @参数=(SELECT 单位1参数 FROM 料件表 WHERE 料号 =@料号)
UPDATE 入库表 SET 数量=数量*@参数,损坏数量=损坏数量*@参数,单位=1 WHERE CURRENT OF CUR END
FETCH NEXT FROM CUR INTO @料号,@单位END
CLOSE CUR
DEALLOCATE CUR
可以改写成:
UPDATE A SET 数量=CASE A.单位 WHEN 1 THEN A.数量*B. 单位1参数
WHEN 2 THEN A.数量*B. 单位2参数 ELSE A.数量
END, 损坏数量= CASE A.单位 WHEN 1 THEN A. 损坏数量*B. 单位1参数
WHEN 2 THEN A. 损坏数量*B. 单位2参数 ELSE A. 损坏数量
END,单位=1
FROM入库表 A, 料件表 BWHERE A.单位<>1 AND
A.料号=B.料号
C、
变量参与的UPDATE语句的例子
SQL ERVER的语句比较灵活,变量参与的UPDATE语句可以实现一些游标一样的功能,比如:

SELECT A,B,C,CAST(NULL AS INT) AS 序号INTO #T
FROM 表
ORDER BY A ,NEWID()
产生临时表后,已经按照A字段排序,但是在A相同的情况下是乱序的,这时如果需要更改序号字段为按照A字段分组的记录序号,就只有游标和变量参与的UPDATE语句可以实现了,这个变量参与的UPDATE语句如下:
DECLARE @A INTDECLARE @序号 INT
UPDATE #T SET @序号=CASE WHEN A=@A THEN @序号+1 ELSE 1 END,
@A=A,
序号=@序号
D、如果必须使用游标,注意选择游标的类型,如果只是循环取数据,那就应该用只进游标(选项FAST_FORWARD),一般只需要静态游标(选项STATIC)。
E、
注意动态游标的不确定性,动态游标查询的记录集数据如果被修改,会自动刷新游标,这样使得动态游标有了不确定性,因为在多用户环境下,如果其他进程或者本身更改了纪录,就可能刷新游标的记录集。

分享到:
评论

相关推荐

    SQLSERVER性能优化综述

    SQLSERVER性能优化综述 在软件开发过程中,性能优化是一个贯穿始终的重要环节,不仅限于测试和维护阶段,也包括从需求分析到设计、开发、部署等各个阶段。本文将根据软件生命周期的不同阶段,探讨数据库性能优化的...

    SQLServer性能优化与管理的艺术

    资源名称:SQL Server性能优化与管理的艺术内容简介...第二部分(第3-10章)为知识准备部分,这部分介绍了SQLServer性能相关的基础知识。只有了解了性能及影 资源太大,传百度网盘了,链接在附件中,有需要的同学自取。

    SqlServer性能优化高效索引指南.pdf

    Sql Server性能优化高效索引指南 Sql Server性能优化高效索引指南是指在Sql Server数据库中,通过合理地设计和优化索引来提高数据库性能的一系列指南和最佳实践。本指南涵盖了索引的基本概念、索引的类型、索引的...

    sql server性能优化(总结)

    SQL Server 性能优化综述 一个系统的性能提高,不单单是试运行或者维护阶段的性能调优的任务,也不单单是开发阶段的事情,而是在整个软件生命周期都需要注意,进行有效工作才能达到的。所以我希望按照软件生命周期...

    SQL SERVER性能优化综述(很好的总结,不要错过哦)第1/3页

    总结来说,SQL Server性能优化是一个全生命周期的工程,需要在系统的分析、设计、实施和维护各个阶段都进行细致的规划和调整。每个阶段都有其特定的优化策略和关注点,只有将这些策略综合运用,才能构建出满足业务...

    SqlServer性能优化

    SqlServer性能优化 SqlServer性能优化是关系数据库管理系统(RDBMS)...SqlServer性能优化是一个复杂的过程,需要从软件生命周期的不同阶段对数据库性能优化的注意事项进行总结和分析,以提高整个系统的性能和可靠性。

    Sql Server——Sql性能优化

    Sql Server——Sql性能优化Sql Server——Sql性能优化Sql Server——Sql性能优化Sql Server——Sql性能优化Sql Server——Sql性能优化Sql Server——Sql性能优化Sql Server——Sql性能优化Sql Server——Sql性能优化

    《SQL Server 2008查询性能优化》扫描版[PDF] 第一卷

    《SQL Server 2008查询性能优化...个人感觉不错,对Sqlserver优化很有帮助,主要都是我们经常犯的错,值得借鉴:如:低性能的查询、索引、死锁等等。 由于网站上传限制及本书大小限制,本书分四卷地址如下: 第一卷: ...

    SQL SERVER性能优化综述

    SQL SERVER性能优化是一个涵盖多个层面的复杂话题,它涉及到数据库设计、索引策略、查询优化、资源管理等多个方面。在确保系统稳定性和数据完整性的同时,提高SQL SERVER的性能至关重要。 首先,性能优化始于分析...

    SqlServer性能优化方面的总结归纳和实战

    在SQL Server性能优化方面,有许多关键点需要理解并掌握,以确保数据库系统高效运行。本文将深入探讨SQL Server性能优化的策略、索引的作用以及优化的阶段。 首先,我们需要了解SQL Server性能优化的重要性。数据库...

    SQL+Server+性能优化及管理艺术 脚本优化文件

    在SQL Server数据库管理系统中,性能优化与管理是至关重要的任务,因为这直接影响到系统的响应速度、资源利用率以及整体系统的稳定性。"SQL Server性能优化及管理艺术"这一主题涵盖了多个方面,包括查询优化、索引...

    sql server 性能优化

    首先,"SQL SERVER性能优化综述.docx"可能是一个全面介绍SQL Server性能调优的文档,涵盖了基础概念、最佳实践以及常见的性能问题解决方案。通常,这样的文档会讨论到如查询优化器的工作原理,如何解读执行计划,...

    SqlServer性能优化高效索引指南

    SqlServer性能优化是数据库管理员和开发人员必须面对的常见任务,而高效索引则是提升SqlServer数据库性能的关键手段之一。索引是数据库表中的一种数据结构,用于加速查询操作,尤其是那些涉及WHERE子句、JOIN操作和...

    SQLServer性能监控指标说明

    SQL Server 的性能监控指标是数据库管理员和开发者对数据库性能进行监控和优化的重要依据。以下是 SQL Server 性能监控指标的说明: 配置硬件 在 SQL Server 中,配置硬件是性能监控的重要方面。硬件配置包括内存...

    SQL Server 性能优化及管理艺术》下载内容(47324).rar

    《SQL Server性能优化及管理艺术》是一本深入探讨SQL Server数据库管理系统性能提升和管理策略的专业书籍。这本书的主要焦点在于如何通过一系列技术手段和最佳实践,让SQL Server运行得更加高效和稳定。书中涵盖了...

    SQL Server 2000完结篇系列之十:SQL Server 2000性能优化答疑

    在SQL Server 2000性能优化答疑这个专题中,我们将深入探讨如何提升数据库系统的运行效率,解决在实际操作中可能遇到的各种性能瓶颈问题。SQL Server 2000是微软公司推出的一款关系型数据库管理系统,尽管现在已经...

    sql server 性能优化(生产中常用)

    NOWAIT是一个与锁相关的选项,它用于指示SQL Server不要等待获取锁,而是立即返回。在并发控制和锁管理中,合理使用NOWAIT可以提高数据库操作的响应速度。 26. NOLOCK使用 NOLOCK是一种锁提示,它允许事务读取未...

    sql server 性能优化大全

    本篇文章将系统地介绍SQL Server性能优化的全方位技术与方案,从底层原理到实际应用。 首先,数据库优化的终极目标包括避免磁盘I/O瓶颈、减少CPU利用率以及减少资源竞争。实现这些目标可以从三个角度进行: 1. ...

    SQL_Server_2008查询性能优化

    在SQL Server 2008查询性能优化的主题中,我们主要关注如何提高SQL查询的速度,减少资源消耗,以及提升数据库的整体效率。对于那些已经具备基本SQL语法知识,并期望提升其代码执行性能的开发者来说,这是一个至关...

Global site tag (gtag.js) - Google Analytics