程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 數據庫知識 >> MYSQL數據庫 >> MySQL綜合教程 >> 詳解MySQL 數據庫優化方法

詳解MySQL 數據庫優化方法

編輯:MySQL綜合教程

用analyze進行處理,定期進行處理

ANALYZE [LOCAL | NO_WRITE_TO_BINLOG] TABLE tb1_name[, tbl_name]...

對表進行定義分析analyze table table_name

CHECK TABLE tb1_name[,tbl_name]...[option]...option = {QUICK | FAST | MEDIUM | EXTENDED | CHANGED}

定期對表進行優化

OPTIMIZE [LOCAL | NO_WRITE_TO_BINLOG] TABLE tb1_name [,tbl_name]...

則應使用OPTIMIZE TABLE命令來進行表優化。這個命令可以將表中的空間碎片進行合並,並且可以消除由於刪除或者更新造成的空間浪費,但OPTIMIZE TABLE 命令只對MyISAM、 BDB 和InnoDB表起作用。

例如: optimize table table_name

下面是mysql教程服務器優化配置的方法

(1)、back_log:
要求 MySQL 能有的連接數量。當主要MySQL線程在一個很短時間內得到非常多的連接請求,這就起作用,然後主線程花些時間(盡管很短)檢查連接並且啟動一個新線程。
back_log值指出在MySQL暫時停止回答新請求之前的短時間內多少個請求可以被存在堆棧中。只有如果期望在一個短時間內有很多連接,你需要增加它,換句話說,這值對到來的TCP/IP連接的偵聽隊列的大小。你的操作系統在這個隊列大小上有它自己的限制。 試圖設定back_log高於你的操作系統的限制將是無效的。
當你觀察你的主機進程列表,發現大量 264084 | unauthenticated user | xxx.xxx.xxx.xxx | NULL | Connect | NULL | login | NULL 的待連接進程時,就要加大 back_log 的值了。默認數值是50,我把它改為500。

(2)、interactive_timeout:
服務器在關閉它前在一個交互連接上等待行動的秒數。一個交互的客戶被定義為對 mysql_real_connect()使用 CLIENT_INTERACTIVE 選項的客戶。 默認數值是28800,我把它改為7200。

(3)、key_buffer_size:
索引塊是緩沖的並且被所有的線程共享。key_buffer_size是用於索引塊的緩沖區大小,增加它可得到更好處理的索引(對所有讀和多重寫),到你能負擔得起那樣多。如果你使它太大,系統將開始換頁並且真的變慢了。默認數值是8388600(8M),我的MySQL主機有2GB內存,所以我把它改為402649088(400MB)。

(4)、max_connections:
允許的同時客戶的數量。增加該值增加 mysqld 要求的文件描述符的數量。這個數字應該增加,否則,你將經常看到 Too many connections 錯誤。 默認數值是100,我把它改為1024 。

(5)、record_buffer:
每個進行一個順序掃描的線程為其掃描的每張表分配這個大小的一個緩沖區。如果你做很多順序掃描,你可能想要增加該值。默認數值是131072(128K),我把它改為16773120 (16M)

(6)、sort_buffer:
每個需要進行排序的線程分配該大小的一個緩沖區。增加這值加速ORDER BY或GROUP BY操作。默認數值是2097144(2M),我把它改為 16777208 (16M)。

(7)、table_cache:
為所有線程打開表的數量。增加該值能增加mysqld要求的文件描述符的數量。MySQL對每個唯一打開的表需要2個文件描述符。默認數值是64,我把它改為512。

(、thread_cache_size:
可以復用的保存在中的線程的數量。如果有,新的線程從緩存中取得,當斷開連接的時候如果有空間,客戶的線置在緩存中。如果有很多新的線程,為了提高性能可以這個變量值。通過比較 Connections 和 Threads_created 狀態的變量,可以看到這個變量的作用。我把它設置為 80。

(10)、wait_timeout:
服務器在關閉它之前在一個連接上等待行動的秒數。 默認數值是28800,我把它改為7200。

注:參數的調整可以通過修改 /etc/my.cnf 文件並重啟 MySQL 實現。這是一個比較謹慎的工作,上面的結果也僅僅是我的一些看法,你可以根據你自己主機的硬件情況(特別是內存大小)進一步修改。

在Apache, PHP, MySQL的體系架構中,MySQL對於性能的影響最大,也是關鍵的核心部分。對於Discuz!論壇程序也是如此,MySQL的設置是否合理優化,直接影響到論壇的速度和承載量!同時,MySQL也是優化難度最大的一個部分,不但需要理解一些MySQL專業知識,同時還需要長時間的觀察統計並且根據經驗進行判斷,然後設置合理的參數。 下面我們了解一下MySQL優化的一些基礎,MySQL的優化我分為兩個部分,一是服務器物理硬件的優化;二是MySQL自身(my.cnf)的優化。

(1) 服務器硬件對MySQL性能的影響
a) 磁盤尋道能力(磁盤I/O),以目前高轉速SCSI硬盤(7200轉/秒)為例,這種硬盤理論上每秒尋道7200次,這是物理特性決定的,沒有辦法改變。MySQL每秒鐘都在進行大量、復雜的查詢操作,對磁盤的讀寫量可想而知。所以,通常認為磁盤I/O是制約MySQL性能的最大因素之一,對於日均訪問量在100萬PV以上的Discuz!論壇,由於磁盤I/O的制約,MySQL的性能會非常低下!解決這一制約因素可以考慮以下幾種解決方案: 使用RAID-0+1磁盤陣列,注意不要嘗試使用RAID-5,MySQL在RAID-5磁盤陣列上的效率不會像你期待的那樣快; 拋棄傳統的硬盤,使用速度更快的閃存式存儲設備。經過Discuz!公司技術工程的測試,使用閃存式存儲設備可比傳統硬盤速度高出6-10倍左右。
b) CPU 對於MySQL應用,推薦使用S.M.P.架構的多路對稱CPU,例如:可以使用兩顆Intel Xeon 3.6GHz的CPU。
c) 物理內存對於一台使用MySQL的Database Server來說,服務器內存建議不要小於2GB,推薦使用4GB以上的物理內存。

(2) MySQL自身因素當解決了上述服務器硬件制約因素後,讓我們看看MySQL自身的優化是如何操作的。對MySQL自身的優化主要是對其配置文件my.cnf中的各項參數進行優化調整。下面我們介紹一些對性能影響較大的參數。 由於my.cnf文件的優化設置是與服務器硬件配置息息相關的,因而我們指定一個假想的服務器硬件環境:
CPU: 2顆Intel Xeon 2.4GHz 內存: 4GB DDR 硬盤: SCSI 73GB
下面,我們根據以上硬件配置結合一份已經優化好的my.cnf進行說明:
# vi /etc/my.cnf以下只列出my.cnf文件中[mysqld]段落中的內容,其他段落內容對MySQL運行性能影響甚微,因而姑且忽略。
[mysqld]
port = 3306
serverid = 1
socket = /tmp/mysql.sock
skip-locking
# 避免MySQL的外部鎖定,減少出錯幾率增強穩定性。
skip-name-resolve禁止MySQL對外部連接進行DNS解析,使用這一選項可以消除MySQL進行DNS解析的時間。但需要注意,如果開啟該選項,則所有遠程主機連接授權都要使用IP地址方式,否則MySQL將無法正常處理連接請求!

 


back_log = 384指定MySQL可能的連接數量。當MySQL主線程在很短的時間內接收到非常多的連接請求,該參數生效,主線程花費很短的時間檢查連接並且啟動一個新線程。
back_log參數的值指出在MySQL暫時停止響應新請求之前的短時間內多少個請求可以被存在堆棧中。 如果系統在一個短時間內有很多連接,則需要增大該參數的值,該參數值指定到來的TCP/IP連接的偵聽隊列的大小。不同的操作系統在這個隊列大小上有它自己的限制。 試圖設定back_log高於你的操作系統的限制將是無效的。默認值為50。對於Linux系統推薦設置為小於512的整數。


key_buffer_size = 256M
# key_buffer_size指定用於索引的緩沖區大小,增加它可得到更好的索引處理性能。對於內存在4GB左右的服務器該參數可設置為256M或384M。注意:該參數值設置的過大反而會是服務器整體效率降低!
max_allowed_packet = 4M
thread_stack = 256K
table_cache = 128K
sort_buffer_size = 6M查詢排序時所能使用的緩沖區大小。注意:該參數對應的分配內存是每連接獨占!如果有100個連接,那麼實際分配的總共排序緩沖區大小為100 × 6 = 600MB。所以,對於內存在4GB左右的服務器推薦設置為6-8M。


read_buffer_size = 4M讀查詢操作所能使用的緩沖區大小。和sort_buffer_size一樣,該參數對應的分配內存也是每連接獨享!


join_buffer_size = 8M聯合查詢操作所能使用的緩沖區大小,和sort_buffer_size一樣,該參數對應的分配內存也是每連接獨享!


myisam_sort_buffer_size = 64M
table_cache = 512
thread_cache_size = 64
query_cache_size = 64M指定MySQL查詢緩沖區的大小。可以通過在MySQL控制台執行以下命令觀察:
# > SHOW VARIABLES LIKE '%query_cache%';
# > SHOW STATUS LIKE 'Qcache%';
# 如果Qcache_lowmem_prunes的值非常大,則表明經常出現緩沖不夠的情況;如果Qcache_hits的值非常大,則表明查詢緩沖使用非常頻繁,如果該值較小反而會影響效率,那麼可以考慮不用查詢緩沖;Qcache_free_blocks,如果該值非常大,則表明緩沖區中碎片很多。

sql優化方法

,EXPLAIN 輸出的結果格式改變了,使得它更適合例如 UNION 語句、子查詢以及派生表的結構。更令人注意的是,它新增了2個字段: id 和 select_type。當你使用早於MySQL 4.1的版本就看不到這些字段了。

  EXPLAIN 結果的每行記錄顯示了每個表的相關信息,每行記錄都包含以下幾個字段:

  id

  本次 SELECT 的標識符。在查詢中每個 SELECT 都有一個順序的數值。

  select_type

  SELECT 的類型,可能會有以下幾種:

  SIMPLE

  簡單的 SELECT (沒有使用 UNION 或子查詢)

  PRIMARY

  最外層的 SELECT。

  UNION

  第二層,在SELECT 之後使用了 UNION 。

  DEPENDENT UNION

  UNION 語句中的第二個 SELECT,依賴於外部子查詢

  SUBQUERY

  子查詢中的第一個 SELECT

  DEPENDENT SUBQUERY

  子查詢中的第一個 SUBQUERY 依賴於外部的子查詢

  DERIVED

  派生表 SELECT(FROM 子句中的子查詢)

  table

  記錄查詢引用的表。

  type

  表連接類型。以下列出了各種不同類型的表連接,依次是從最好的到最差的:

  system

  表只有一行記錄(等於系統表)。這是 const 表連接類型的一個特例。

  const

  表中最多只有一行匹配的記錄,它在查詢一開始的時候就會被讀取出來。由於只有一行記錄,在余下的優化程序裡該行記錄的字段值可以被當作是一個恆定值。const 表查詢起來非常快,因為只要讀取一次!const 用於在和 PRIMARY KEY 或 UNIQUE 索引中有固定值比較的情形。下面的幾個查詢中,tbl_name 就是 const 表了:

  
  SELECT*FROMtbl_nameWHEREprimary_key=1;
  SELECT*FROMtbl_name
  WHEREprimary_key_part1=1ANDprimary_key_part2=2;
 


  eq_ref

  從該表中會有一行記錄被讀取出來以和從前一個表中讀取出來的記錄做聯合。與 const 類型不同的是,這是最好的連接類型。它用在索引所有部分都用於做連接並且這個索引是一個 PRIMARY KEY 或 UNIQUE 類型。eq_ref 可以用於在進行"="做比較時檢索字段。比較的值可以是固定值或者是表達式,表達式中可以使用表裡的字段,它們在讀表之前已經准備好了。以下的幾個例子中,MySQL使用了 eq_ref 連接來處理 ref_table:

  
  SELECT*FROMref_table,other_table
  WHEREref_table.key_column=other_table.column;
  SELECT*FROMref_table,other_table
  WHEREref_table.key_column_part1=other_table.column
  ANDref_table.key_column_part2=1;
 


  ref

  該表中所有符合檢索值的記錄都會被取出來和從上一個表中取出來的記錄作聯合。ref 用於連接程序使用鍵的最左前綴或者是該鍵不是 PRIMARY KEY 或 UNIQUE 索引(換句話說,就是連接程序無法根據鍵值只取得一條記錄)的情況。當根據鍵值只查詢到少數幾條匹配的記錄時,這就是一個不錯的連接類型。ref 還可以用於檢索字段使用 = 操作符來比較的時候。以下的幾個例子中,MySQL將使用 ref 來處理 ref_table:

  
  SELECT*FROMref_tableWHEREkey_column=expr;
  SELECT*FROMref_table,other_table
  WHEREref_table.key_column=other_table.column;
  SELECT*FROMref_table,other_table
  WHEREref_table.key_column_part1=other_table.column
  ANDref_table.key_column_part2=1;


  ref_or_null

  這種連接類型類似 ref,不同的是MySQL會在檢索的時候額外的搜索包含 NULL 值的記錄。這種連接類型的優化是從MySQL 4.1.1開始的,它經常用於子查詢。在以下的例子中,MySQL使用 ref_or_null 類型來處理 ref_table:

  
  SELECT*FROMref_table
  WHEREkey_column=exprORkey_columnISNULL;


  index_merge

  這種連接類型意味著使用了 Index Merge 優化方法。這種情況下,key字段包括了所有使用的索引,key_len 包括了使用的鍵的最長部分。詳情請看"7.2.5 How MySQL Optimizes OR Clauses"。

  unique_subquery

  這種類型用例如一下形式的 IN 子查詢來替換 ref:

  value IN (SELECT primary_key FROM single_table WHERE some_expr)

  unique_subquery 只是用來完全替換子查詢的索引查找函數效率更高了。

  index_subquery

  這種連接類型類似 unique_subquery。它用子查詢來代替 IN,不過它用於在子查詢中沒有唯一索引的情況下,例如以下形式:

  value IN (SELECT key_column FROM single_table WHERE some_expr)

  range

  只有在給定范圍的記錄才會被取出來,利用索引來取得一條記錄。key 字段表示使用了哪個索引。key_len 字段包括了使用的鍵的最長部分。這種類型時 ref 字段值是 NULL。range 用於將某個字段和一個定植用以下任何操作符比較時 =, <>, >, >=, <, <=, IS NULL, <=>, BETWEEN, 或 IN:
tmp_table_size = 256M
max_connections = 768指定MySQL允許的最大連接進程數。如果在訪問論壇時經常出現Too Many Connections的錯誤提 示,則需要增大該參數值。


max_connect_errors = 10000000
wait_timeout = 10指定一個請求的最大連接時間,對於4GB左右內存的服務器可以設置為5-10。


thread_concurrency = 8該參數取值為服務器邏輯CPU數量×2,在本例中,服務器有2顆物理CPU,而每顆物理CPU又支持H.T超線程,所以實際取值為4 × 2 = 8


skip-networking開啟該選項可以徹底關閉MySQL的TCP/IP連接方式,如果WEB服務器是以遠程連接的方式訪問MySQL數據庫教程服務器則不要開啟該選項!否則將無法正常連接!

 

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved