1、首先要搞明白什么叫执行计划?

  执行计划是数据库根据SQL语句和相关表的统计信息作出的一个查询方案,这个方案是由查询优化器自动分析产生的,比如一条SQL语句如果用来从一个10万条记录的表中查1条记录,那查询优化器会选择“索引查找”方式,如果该表进行了归档,当前只剩下5000条记录了,那查询优化器就会改变方案,采用“全表扫描”方式。

  可见,执行计划并不是固定的,它是“个性化的”。产生一个正确的“执行计划”有两点很重要:

  (1) SQL语句是否清晰地告诉查询优化器它想干什么?

  (2) 查询优化器得到的数据库统计信息是否是最新的、正确的?

  2、统一SQL语句的写法

  对于以下两句SQL语句,程序员认为是相同的,数据库查询优化器认为是不同的。

  select * from dual

  Select * From dual

  其实就是大小写不同,查询分析器就认为是两句不同的SQL语句,必须进行两次解析。生成2个执行计划。所以作为程序员,应该保证相同的查询语句在任何地方都一致,多一个空格都不行!

  3、不要把SQL语句写得太复杂

  我经常看到,从数据库中捕捉到的一条SQL语句打印出来有2张A4纸这么长。一般来说这么复杂的语句通常都是有问题的。我拿着这2页长的SQL语句去请教原作者,结果他说时间太长,他一时也看不懂了。可想而知,连原作者都有可能看糊涂的SQL语句,数据库也一样会看糊涂。

  一般,将一个SELECT语句的结果作为子集,然后从该子集中再进行查询,这种一层嵌套语句还是比较常见的,但是根据经验,超过3层嵌套,查询优化器就很容易给出错误的执行计划。因为它被绕晕了。像这种类似人工智能的东西,终究比人的分辨力要差些,如果人都看晕了,我可以保证数据库也会晕的。

  另外,执行计划是可以被重用的,越简单的SQL语句被重用的可能性越高。而复杂的SQL语句只要有一个字符发生变化就必须重新解析,然后再把这一大堆垃圾塞在内存里。可想而知,数据库的效率会何等低下。

  4、使用“临时表”暂存中间结果

  简化SQL语句的重要方法就是采用临时表暂存中间结果,但是,临时表的好处远远不止这些,将临时结果暂存在临时表,后面的查询就在TempDB中了,这可以避免程序中多次扫描主表,也大大减少了程序执行中“共享锁”阻塞“更新锁”,减少了阻塞,提高了并发性能。

  5、OLTP系统SQL语句必须采用绑定变量

  select * from orderheader where changetime > ‘2010-10-20 00:00:01’

  select * from orderheader where changetime > ‘2010-09-22 00:00:01’

  以上两句语句,查询优化器认为是不同的SQL语句,需要解析两次。如果采用绑定变量

  select * from orderheader where changetime > @chgtime

  @chgtime变量可以传入任何值,这样大量的类似查询可以重用该执行计划了,这可以大大降低数据库解析SQL语句的负担。一次解析,多次重用,是提高数据库效率的原则。

  6、绑定变量窥测

  事物都存在两面性,绑定变量对大多数OLTP处理是适用的,但是也有例外。比如在WHERE条件中的字段是“倾斜字段”的时候。

  “倾斜字段”指该列中的绝大多数的值都是相同的,比如一张人口调查表,其中“民族”这列,90%以上都是汉族。那么如果一个SQL语句要查询30岁的汉族人口有多少,那“民族”这列必然要被放在WHERE条件中。这个时候如果采用绑定变量@nation会存在很大问题。

  试想如果@nation传入的第一个值是“汉族”,那整个执行计划必然会选择表扫描。然后,第二个值传入的是“布依族”,按理说“布依族”占的比例可能只有万分之一,应该采用索引查找。但是,由于重用了第一次解析的“汉族”的那个执行计划,那么第二次也将采用表扫描方式。这个问题就是著名的“绑定变量窥测”,建议对于“倾斜字段”不要采用绑定变量。

  7、只在必要的情况下才使用BEGIN TRAN

  SQL Server中一句SQL语句默认就是一个事务,在该语句执行完成后也是默认COMMIT的。其实,这就是BEGIN TRAN的一个最小化的形式,好比在每句语句开头隐含了一个BEGIN TRAN,结束时隐含了一个COMMIT。

  有些情况下,我们需要显式声明BEGIN TRAN,比如做“插、删、改”操作需要同时修改几个表,要求要么几个表都修改成功,要么都不成功。BEGIN TRAN可以起到这样的作用,它可以把若干SQL语句套在一起执行,最后再一起COMMIT。好处是保证了数据的一致性,但任何事情都不是完美无缺的。BEGIN TRAN付出的代价是在提交之前,所有SQL语句锁住的资源都不能释放,直到COMMIT掉。

  可见,如果BEGIN TRAN套住的SQL语句太多,那数据库的性能就糟糕了。在该大事务提交之前,必然会阻塞别的语句,造成BLOCK很多。

  BEGIN TRAN使用的原则是,在保证数据一致性的前提下,BEGIN TRAN套住的SQL语句越少越好!有些情况下可以采用触发器同步数据,不一定要用BEGIN TRAN。

  8、一些SQL查询语句应加上NOLOCK

  在SQL语句中加NOLOCK是提高SQL Server并发性能的重要手段,在Oracle中并不需要这样做,因为Oracle的结构更为合理,有UNDO表空间保存“数据前影”,该数据如果在修改中还未COMMIT,那么你读到的是它修改之前的副本,该副本放在UNDO表空间中。这样,Oracle的读、写可以做到互不影响,这也是Oracle广受称赞的地方。SQL Server的读、写是会相互阻塞的,为了提高并发性能,对于一些查询,可以加上NOLOCK,这样读的时候可以允许写,但缺点是可能读到未提交的脏数据。使用NOLOCK有3条原则。

  (1) 查询的结果用于“插、删、改”的不能加NOLOCK!

  (2) 查询的表属于频繁发生页分裂的,慎用NOLOCK!

  (3) 使用临时表一样可以保存“数据前影”,起到类似Oracle的UNDO表空间的功能,能采用临时表提高并发性能的,不要用NOLOCK!

  9、聚集索引没有建在表的顺序字段上,该表容易发生页分裂

  比如订单表,有订单编号OrderID,也有客户编号ContactID,那么聚集索引应该加在哪个字段上呢?对于该表,订单编号是顺序添加的,如果在OrderID上加聚集索引,新增的行都是添加在末尾,这样不容易经常产生页分裂。然而,由于大多数查询都是根据客户编号来查的,因此,将聚集索引加在ContactID上才有意义。而ContactID对于订单表而言,并非顺序字段。

  比如“张三”的“contactid”是001,那么“张三”的订单信息必须都放在这张表的第一个数据页上,如果今天“张三”新下了一个订单,那该订单信息不能放在表的最后一页,而是第一页!如果第一页放满了呢?很抱歉,该表所有数据都要往后移动为这条记录腾地方。

  SQL Server的索引和Oracle的索引是不同的,SQL Server的聚集索引实际上是对表按照聚集索引字段的顺序进行了排序,相当于Oracle的索引组织表。SQL Server的聚集索引就是表本身的一种组织形式,所以它的效率是非常高的。也正因为此,插入一条记录,它的位置不是随便放的,而是要按照顺序放在该放的数据页,如果那个数据页没有空间了,就引起了页分裂。所以很显然,聚集索引没有建在表的顺序字段上,该表容易发生页分裂。

  曾经碰到过一个情况,一位哥们的某张表重建索引后,插入的效率大幅下降了。估计情况大概是这样的。该表的聚集索引可能没有建在表的顺序字段上,该表经常被归档,所以该表的数据是以一种稀疏状态存在的。比如张三下过20张订单,而最近3个月的订单只有5张,归档策略是保留3个月数据,那么张三过去的15张订单已经被归档,留下15个空位,可以在INSERT发生时重新被利用。在这种情况下由于有空位可以利用,就不会发生页分裂。但是查询性能会比较低,因为查询时必须扫描那些没有数据的空位。

  重建聚集索引后情况改变了,因为重建聚集索引就是把表中的数据重新排列一遍,原来的空位没有了,而页的填充率又很高,插入数据经常要发生页分裂,所以性能大幅下降。

  对于聚集索引没有建在顺序字段上的表,是否要给与比较低的页填充率?是否要避免重建聚集索引?是一个值得考虑的问题!

  10、加NOLOCK后查询经常发生页分裂的表,容易产生跳读或重复读

  加NOLOCK后可以在“插、删、改”的同时进行查询,但是由于同时发生“插、删、改”,在某些情况下,一旦该数据页满了,那么页分裂不可避免,而此时NOLOCK的查询正在发生,比如在第100页已经读过的记录,可能会因为页分裂而分到第101页,这有可能使得NOLOCK查询在读101页时重复读到该条数据,产生“重复读”。同理,如果在100页上的数据还没被读到就分到99页去了,那NOLOCK查询有可能会漏过该记录,产生“跳读”。

  上面提到的哥们,在加了NOLOCK后一些操作出现报错,估计有可能因为NOLOCK查询产生了重复读,2条相同的记录去插入别的表,当然会发生主键冲突。

  11、使用LIKE进行模糊查询时应注意

  有的时候会需要进行一些模糊查询比如

  select * from contact where username like ‘%yue%’

  关键词%yue%,由于yue前面用到了“%”,因此该查询必然走全表扫描,除非必要,否则不要在关键词前加%,

  12、数据类型的隐式转换对查询效率的影响

  SQL Server 2000的数据库,我们的程序在提交SQL语句的时候,没有使用强类型提交这个字段的值,由SQL Server 2000自动转换数据类型,会导致传入的参数与主键字段类型不一致,这个时候SQL Server 2000可能就会使用全表扫描。SQL Server 2005上没有发现这种问题,但是还是应该注意一下。

  13、SQL Server表连接的三种方式

  (1) MERGE JOIN

  (2) NESTED LOOP JOIN

  (3) HASH JOIN

  SQL Server 2000只有一种JOIN方式——NESTED LOOP JOIN,如果A结果集较小,那就默认作为外表,A中每条记录都要去B中扫描一遍,实际扫过的行数相当于A结果集行数×B结果集行数。所以如果两个结果集都很大,那JOIN的结果很糟糕。

  SQL Server 2005新增了MERGE JOIN,如果A表和B表的连接字段正好是聚集索引所在字段,那么表的顺序已经排好,只要两边拼上去就行了,这种JOIN的开销相当于A表的结果集行数加上B表的结果集行数,一个是加,一个是乘,可见MERGE JOIN的效果要比NESTED LOOP JOIN好多了。

  如果连接的字段上没有索引,那SQL Server 2000的效率是相当低的,而SQL Server 2005提供了HASH JOIN,相当于临时给A,B表的结果集加上索引,因此SQL Server 2005的效率比SQL Server 2000有很大提高,我认为,这是一个重要的原因。

  总结一下,在表连接时要注意以下几点:

  (1) 连接字段尽量选择聚集索引所在的字段;

  (2) 仔细考虑WHERE条件,尽量减小A、B表的结果集;

  (3) 如果很多JOIN的连接字段都缺少索引,而你还在用SQL Server 2000,赶紧升级吧!

转载于:https://www.cnblogs.com/qanholas/archive/2011/06/09/2076342.html

在程序开发中怎样写SQL语句可以提高数据库的性能相关推荐

  1. 【8008】解决idea中JDBC写sql语句出现报错Unable to resolve table ‘表名‘?

    [8008]解决idea中写sql语句出现报错Unable to resolve table '表名'? 问题背景: 昨天使用JDBC连接数据库中写sql语句,一直爆红,真服了,虽然能运行,但看着不舒 ...

  2. java开发中常用的SQL语句

    在实际开发过程中常用的sql语句, 以user_table为例. 1)创建语句如下: CREATE TABLE USER_TABLE (USER_ID VARCHAR2(8), USER_NAME V ...

  3. C++ 开发中如何利用sql语句(insert语句)向数据库中插入变量

    **问题描述:**在C/S开发过程中,当客户端向服务端发送接口(结构体)信息时,服务端需要将接收到的信息存入数据库,但是,常规insert语句只能存确定的值到数据,当客户端传入的值不确定,且为了保证代 ...

  4. sql语句ding_Navicat写sql语句的具体方法步骤

    你们知道Navicat中怎么写sql语句吗?不清楚的朋友可以去下文学习一下Navicat写sql语句的具体方法步骤哦. Navicat写sql语句的具体方法步骤 步骤一 打开你电脑里的Navicat ...

  5. 程序员老鸟写sql语句的经验之谈

    程序员老鸟写sql语句的经验之谈 做管理系统的,无论是bs结构的还是cs结构的,都不可避免的涉及到数据库表结构的设计,sql语句的编写等.因此在开发系统的时候,表结构设计是否合理,sql语句是否标准, ...

  6. java sql范围查询语句,java类中写sql语句,查询条件包含换行

    java类中写sql语句,查询条件包含换行 detachedCriteria.add(Restrictions.or( Restrictions.like("chengBanDanWeiId ...

  7. java mysql查询字段换行,java类中写sql语句,查询条件包含换行

    java类中写sql语句,查询条件包含换行 detachedCriteria.add(Restrictions.or( Restrictions.like("chengBanDanWeiId ...

  8. oracle中执行某sql语句后,如一系列delete、update等操作,怎么看影响的行数?如我执行了n个表的delete语句,得看我删除了多少数据,万一sql写的有问题,误删了数据,不是灾难?

    oracle中执行某sql语句后,如一系列delete.update等操作,怎么看影响的行数?如我执行了n个表的delete语句,得看我删除了多少数据,万一sql写的有问题,误删了数据,不是灾难? 使 ...

  9. ACCESS中写SQL语句时尽量把表名和字段名用中括号括起来

    ACCESS中写SQL语句时尽量把表名和字段名用中括号括起来,一般都要写成: [TableName].FieldName,否则较长的中文名表名可能会被不识别.

最新文章

  1. 无法从“std::_Binder std::_Unforced,SOCKET ,LPSOCKADDR,unsigned int ”转换为“int”
  2. XenApp_XenDesktop_7.6实战篇之十五:StoreFront的配置
  3. SIM: 基于搜索的超长行为序列上的用户兴趣建模
  4. .net前台ajax,asp.net利用Ajax和Jquery在前台向后台传参数并返回值
  5. php运行环境出现Undefined index 或variable时解决方法
  6. python (集合和深浅拷贝)
  7. DenseNet翻译:Densely Connected Convolutional Networks
  8. DreamweaverCS6手把手教你安装并破解
  9. 计算机视觉论文-2021-09-07
  10. 关于git push报hook declined
  11. 域控服务器的ip地址,修改Active Directory域控制器IP地址
  12. hdfs 指令_HDFS之一:hdfs命令行操作
  13. php获取企业微信聊天内容,微信企业号开发获取用户信息的介绍
  14. 创建会计凭证BAPI_ACC_DOCUMENT_CHECK 原因代码和凭证类型必须增强
  15. 使用opencv截取旋转框目标
  16. C# Xamarin For Android移动开发项目实战篇
  17. 密码的修改(首先获取该用户的id,原密码判断是否一致,新密码和再次输入密码判断是否一样)...
  18. 表情识别(二)--基于CNN分类
  19. 查找磁盘占用最大的文件linux,linux查找系统中占用磁盘空间最大的文件
  20. 安卓开发——在应用中简单调用Android百度地图API SDK

热门文章

  1. 微信新功能能够给企业、个人商家带来什么?
  2. 踩了无数个坑,才写出稿费1000的文章
  3. 今年的大环境很差,创业失败的和失去工作的特别多
  4. 如今在农村做生意,为啥越来越难了?
  5. 什么叫做数字功放?它的电路原理是什么?
  6. 易懂分布式 | Kademlia算法
  7. binary.Write 小坑一个兼论go的错误处理哲学
  8. Linux上SQL Server事务复制
  9. sql 自定义函数 示例_SQL Server SESSION_CONTEXT()函数与示例
  10. Spring注解@ConfigurationPropertie