首頁 資料庫 mysql教程 SQL高性能查询优化语句(总结)_MySQL

SQL高性能查询优化语句(总结)_MySQL

May 27, 2016 pm 07:11 PM
語句 高效能

SQL 高性能查询优化语句,一些经验总结

 

1.应尽量避免在 where 子句中对字段进行 null 值判断,否则将导致引擎放弃使用索引而进行全表扫描,如:

select id from t where num is null;

可以在num上设置默认值0,确保表中num列没有null值,然后这样查询:

 

select id from t where num = 0;

 

2.应尽量避免在 where 子句中使用!=或$amp;

 

3.应尽量避免在 where 子句中使用 or 来连接条件,否则将导致引擎放弃使用索引而进行全表扫描,如:

select id from t where num=10 or num=20;

可以这样查询:

 

select id from t where num=10

union all

select id from t where num=20;

 

4.in 和 not in 也要慎用,因为IN会使系统无法使用索引,而只能直接搜索表中的数据。如:

select id from t where num in(1,2,3)

 

对于连续的数值,能用 between 就不要用 in 了:

 

select id from t where num between 1 and 3;

 

5.尽量避免在索引过的字符数据中,使用非打头字母搜索。这也使得引擎无法利用索引。

 

见如下例子:

 

SELECT * FROM T1 WHERE NAME LIKE ‘%L%’;

SELECT * FROM T1 WHERE SUBSTING(NAME,2,1)=’L’;

SELECT * FROM T1 WHERE NAME LIKE ‘L%’;   --第三个查询能够使用索引来加快操作

 

即使NAME字段建有索引,前两个查询依然无法利用索引完成加快操作,引擎不得不对全表所有数据逐条操作来完成任务。而第三个查询能够使用索引来加快操作。

 

 

6.必要时强制查询优化器使用某个索引,如在 where 子句中使用参数,也会导致全表扫描。因为SQL只有在运行时才会解析局部变量,但优化程序不能将访问计划的选择推迟到运行时;它必须在编译时进行选择。然而,如果在编译时建立访问计划,变量的值还是未知的,因而无法作为索引选择的输入项。如下面语句将进行全表扫描:

select id from t where num=@num;

 

可以改为强制查询使用索引:

 

select id from t with(index(索引名)) where num=@num;

 

7.应尽量避免在 where 子句中对字段进行表达式操作,这将导致引擎放弃使用索引而进行全表扫描。如:

SELECT * FROM T1 WHERE F1/2=100;

应改为:

 

SELECT * FROM T1 WHERE F1=100*2;

SELECT * FROM RECORD WHERE SUBSTRING(CARD_NO,1,4)=’5378’;

应改为:

 

SELECT * FROM RECORD WHERE CARD_NO LIKE ‘5378%’;

SELECT member_number, first_name, last_name FROM members

WHERE DATEDIFF(yy,datofbirth,GETDATE()) > 21;

应改为:

 

SELECT member_number, first_name, last_name FROM members

    WHERE dateofbirth

 

即:任何对列的操作都将导致表扫描,它包括数据库函数、计算表达式等等,查询时要尽可能将操作移至等号右边。

 

 

8.应尽量避免在where子句中对字段进行函数操作,这将导致引擎放弃使用索引而进行全表扫描。如:

select id from t where substring(name,1,3)='abc';--name 以abc开头的id

select id from t where datediff(day,createdate,'2005-11-30')=0;--‘2005-11-30’生成的id

 

应改为:

 

select id from t where name like 'abc%';

select id from t where createdate>=’2005-11-30′ and createdate$amp;

 

9.不要在 where 子句中的“=”左边进行函数、算术运算或其他表达式运算,否则系统将可能无法正确使用索引。

 

10.在使用索引字段作为条件时,如果该索引是复合索引,那么必须使用到该索引中的第一个字段作为条件时才能保证系统使用该索引,否则该索引将不会被使用,并且应尽可能的让字段顺序与索引顺序相一致。

 

11.很多时候用 exists是一个好的选择:

select num from a where num in(select num from b);

用下面的语句替换:

 

select num from a where exists(select 1 from b where num=a.num);

再例如:

 

SELECT SUM(T1.C1) FROM T1 WHERE (

SELECT COUNT(*) FROM T2 WHERE T2.C2=T1.C2)>0;

 

SELECT SUM(T1.C1) FROM T1 WHERE EXISTS(

SELECT * FROM T2 WHERE T2.C2=T1.C2);  --效率高

 

两者产生相同的结果,但是后者的效率显然要高于前者。因为后者不会产生大量锁定的表扫描或是索引扫描。

 

如果你想校验表里是否存在某条纪录,不要用count(*)那样效率很低,而且浪费服务器资源。可以用EXISTS代替。如:

IF (SELECT COUNT(*) FROM table_name WHERE column_name = ‘xxx’);

可以写成:

 

IF EXISTS (SELECT * FROM table_name WHERE column_name = 'xxx');

 

经常需要写一个T_SQL语句比较一个父结果集和子结果集,从而找到是否存在在父结果集中有而在子结果集中没有的记录,如:

 

SELECT a.hdr_key FROM hdr_tbl a--hdr_tbl a 表示tbl用别名a代替

WHERE NOT EXISTS (SELECT * FROM dtl_tbl b WHERE a.hdr_key = b.hdr_key)

 

SELECT a.hdr_key FROM hdr_tbl a

LEFT JOIN dtl_tbl b ON a.hdr_key = b.hdr_key WHERE b.hdr_key IS NULL

 

SELECT hdr_key FROM hdr_tbl

WHERE hdr_key NOT IN (SELECT hdr_key FROM dtl_tbl)

 

三种写法都可以得到同样正确的结果,但是效率依次降低。

 

 

12.尽量使用表变量来代替临时表。如果表变量包含大量数据,请注意索引非常有限(只有主键索引)。

 

13.避免频繁创建和删除临时表,以减少系统表资源的消耗。

 

14.临时表并不是不可使用,适当地使用它们可以使某些例程更有效,例如,当需要重复引用大型表或常用表中的某个数据集时。但是,对于一次性事件,最好使用导出表。

 

15.在新建临时表时,如果一次性插入数据量很大,那么可以使用 select into 代替 create table,避免造成大量 log ,以提高速度;如果数据量不大,为了缓和系统表的资源,应先create table,然后insert。

 

16.如果使用到了临时表,在存储过程的最后务必将所有的临时表显式删除,先 truncate table ,然后 drop table ,这样可以避免系统表的较长时间锁定。

 

17.在所有的存储过程和触发器的开始处设置 SET NOCOUNT ON ,在结束时设置 SET NOCOUNT OFF 。无需在执行存储过程和触发器的每个语句后向客户端发送 DONE_IN_PROC 消息。

 

18.尽量避免大事务操作,提高系统并发能力。

 

19.尽量避免向客户端返回大数据量,若数据量过大,应该考虑相应需求是否合理。

20.避免使用不兼容的数据类型。例如float和int、char和varchar、binary和varbinary是不兼容的。数据类型的不兼容可能使优化器无法执行一些本来可以进行的优化操作。例如:

 

SELECT name FROM employee WHERE salary > 60000;

 

在这条语句中,如salary字段是money型的,则优化器很难对其进行优化,因为60000是个整型数。我们应当在编程时将整型转化成为钱币型,而不要等到运行时转化。

 

21.充分利用连接条件,在某种情况下,两个表之间可能不只一个的连接条件,这时在 WHERE 子句中将连接条件完整的写上,有可能大大提高查询速度。

例:

 

SELECT SUM(A.AMOUNT) FROM ACCOUNT A,CARD B WHERE A.CARD_NO = B.CARD_NO;

SELECT SUM(A.AMOUNT) FROM ACCOUNT A,CARD B WHERE A.CARD_NO = B.CARD_NO AND A.ACCOUNT_NO = B.ACCOUNT_NO;--完整的表链接条件有可能大大提高查询速度

第二句将比第一句执行快得多。

 

22、使用视图加速查询

 

把表的一个子集进行排序并创建视图,有时能加速查询。它有助于避免多重排序 操作,而且在其他方面还能简化优化器的工作。例如:

 

SELECT cust.name,rcvbles.balance,……other columns

FROM cust,rcvbles

WHERE cust.customer_id = rcvlbes.customer_id

    AND rcvblls.balance>0

    AND cust.postcode>“98000”

ORDER BY cust.name;

 

如果这个查询要被执行多次而不止一次,可以把所有未付款的客户找出来放在一个视图中,并按客户的名字进行排序:

 

 

CREATE VIEW DBO.V_CUST_RCVLBES    --创建视图   

AS

    SELECT cust.name,rcvbles.balance,……other columns

    FROM cust,rcvbles

    WHERE cust.customer_id = rcvlbes.customer_id

        AND rcvblls.balance>0

    ORDER BY cust.name;

 

然后以下面的方式在视图中查询:

 

SELECT * FROM V_CUST_RCVLBES

WHERE postcode>“98000”;

 

视图中的行要比主表中的行少,而且物理顺序就是所要求的顺序,减少了磁盘I/O,所以查询工作量可以得到大幅减少。

 

 

23、能用DISTINCT的就不用GROUP BY

 

SELECT OrderID FROM Details WHERE UnitPrice > 10 GROUP BY OrderID;

可改为:

SELECT DISTINCT OrderID FROM Details WHERE UnitPrice > 10;

 

24.能用UNION ALL就不要用UNION。

 

UNION ALL不执行SELECT DISTINCT函数,这样就会减少很多不必要的资源;

 

35.尽量不要用SELECT INTO语句。

 

SELECT INOT 语句会导致表锁定,阻止其他用户访问该表。

 

上面提到的是一些基本的提高查询速度的注意事项,但是在更多的情况下,往往需要反复试验比较不同的语句以得到最佳方案。最好的方法当然是测试,看实现相同功能的SQL语句哪个执行时间最少,但是数据库中如果数据量很少,是比较不出来的,这时可以用查看执行计划,即:把实现相同功能的多条SQL语句考到查询分析器,按CTRL+L看查所利用的索引,表扫描次数(这两个对性能影响最大),总体上看询成本百分比即可。

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

熱門話題

Java教學
1664
14
CakePHP 教程
1423
52
Laravel 教程
1317
25
PHP教程
1268
29
C# 教程
1246
24
如何使用Swoole實現高效能的HTTP反向代理伺服器 如何使用Swoole實現高效能的HTTP反向代理伺服器 Nov 07, 2023 am 08:18 AM

如何使用Swoole實現高效能的HTTP反向代理伺服器Swoole是一款基於PHP語言的高效能、非同步、並發的網路通訊框架。它提供了一系列的網路功能,可以用來實作HTTP伺服器、WebSocket伺服器等。在本文中,我們將介紹如何使用Swoole來實作一個高效能的HTTP反向代理伺服器,並提供具體的程式碼範例。環境配置首先,我們需要在伺服器上安裝Swoole擴展

PHP和WebSocket: 建立高效能的即時應用程式 PHP和WebSocket: 建立高效能的即時應用程式 Dec 17, 2023 pm 12:58 PM

PHP和WebSocket:建立高效能的即時應用程式隨著網路的發展和用戶需求的提升,即時應用程式變得越來越普遍。而傳統的HTTP協定在處理即時資料時會有一些限制,例如需要頻繁的輪詢或長輪詢方式來取得最新的資料。為了解決這個問題,WebSocket應運而生。 WebSocket是一種先進的通訊協議,它提供了雙向通訊的能力,允許瀏覽器和伺服器之間即時發送和接

C++高效能程式設計技巧:優化程式碼以應對大規模資料處理 C++高效能程式設計技巧:優化程式碼以應對大規模資料處理 Nov 27, 2023 am 08:29 AM

C++是一種高效能的程式語言,可以為開發人員提供靈活性和可擴充性。尤其在大規模資料處理場景下,C++的高效率和快速運算速度是非常重要的。本文將介紹一些最佳化C++程式碼的技巧,以因應大規模資料處理需求。使用STL容器取代傳統數組在C++程式設計中,數組是常用的資料結構之一。但是,在大規模資料處理中,使用STL容器,如vector,deque,list和set等,可以更

使用Go語言開發實現高效能的語音辨識應用 使用Go語言開發實現高效能的語音辨識應用 Nov 20, 2023 am 08:11 AM

隨著科技的不斷發展,語音辨識技術也得到了長足的進步與應用。語音辨識應用廣泛運用在語音助理、智慧音箱、虛擬實境等領域,為人們提供了更便利和智慧的互動方式。而如何實現高效能的語音辨識應用,則成為了一個值得探討的問題。近年來,Go語言作為一種高效能的程式語言,在語音辨識應用的開發中備受矚目。 Go語言具備並發性高、編寫簡潔、執行速度快等特點,非常適合用於建構高效能

使用Go語言開發實現高效能的人臉辨識應用 使用Go語言開發實現高效能的人臉辨識應用 Nov 20, 2023 am 09:48 AM

使用Go語言開發實現高效能的人臉辨識應用摘要:人臉辨識技術是一個在當今網路時代非常流行的應用領域。本文介紹了使用Go語言開發高效能人臉辨識應用的步驟和流程。透過使用Go語言的並發、高效能和簡單易用的特點,開發人員可以更便捷地建立高效能的人臉辨識應用。引言:在當今的資訊化社會,人臉辨識技術被廣泛應用於安全監控、人臉支付、人臉解鎖等領域。隨著網路的快速發

Docker和Spring Boot的技術實踐:快速建立高效能的應用服務 Docker和Spring Boot的技術實踐:快速建立高效能的應用服務 Oct 21, 2023 am 08:18 AM

Docker和SpringBoot的技術實踐:快速建立高效能的應用服務引言:在當今的資訊時代,網路應用的開發和部署變得越來越重要。隨著雲端運算和虛擬化技術的快速發展,Docker作為一個輕量級的容器技術,受到了廣泛關注和應用。而SpringBoot作為快速開發和部署Java應用的框架,也得到了廣泛認可。本文將探討如何結合Docker和SpringB

打造高效能Python程式設計工作站的電腦配置建議 打造高效能Python程式設計工作站的電腦配置建議 Mar 25, 2024 pm 07:12 PM

標題:打造高效能Python程式設計工作站的電腦配置建議隨著Python語言在資料分析、人工智慧等領域的廣泛應用,越來越多的開發者和研究人員對於建構高效能的Python程式設計工作站需求日益增加。在選擇電腦配置時,除了效能方面的考量外,還應該根據Python編程的特性進行最佳化,以提高程式效率和運行速度。本文將介紹如何打造一台高效能的Python程式設計工作站,並提供具體

如何設計一個高效能的MySQL表結構來實現推薦電影功能? 如何設計一個高效能的MySQL表結構來實現推薦電影功能? Oct 31, 2023 am 09:08 AM

如何設計一個高效能的MySQL表結構來實現推薦電影功能?近年來,推薦系統在電商、社交網路、音樂和影視等領域得到了廣泛應用。其中,推薦電影功能在視訊串流平台上尤其重要。為了實現高性能的推薦電影功能,設計一個合理的MySQL表結構是至關重要的。本文將詳細介紹如何設計一個高效能的MySQL表結構來實現推薦電影功能,並提供程式碼範例。一、需求分析在開始設計表結構之前

See all articles