程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 數據庫知識 >> MYSQL數據庫 >> 關於MYSQL數據庫 >> 設置MySQL中的數據類型來優化運行速度的實例

設置MySQL中的數據類型來優化運行速度的實例

編輯:關於MYSQL數據庫

今天看了一個優化案例覺的挺有代表性,這裡記錄下來做一個標記,來紀念一下隨便的字段定義的問題。

回憶一下,在表的設計中很多人習慣的把表的結構設計成Varchar(64),Varchar(255)之類的,雖然大多數情況只存了5-15個字節.那麼我看一下下面這個案例.
查詢語句:
 

 SELECT SQL_NO_CACHE channel, COUNT(channel) AS visitors FROM xxx_sources WHERE client_id = 1301 GROUP BY client_id, channel;

該表(client_id,channel)是一個組合索引.
利用explain,看一下執行計劃,對於索引使用上看上非常完美
 

mysql> explain SELECT SQL_NO_CACHE channel, COUNT(channel) AS visitors FROM xxx_sources WHERE client_id = 1301 GROUP BY client_id, channel;
+----+-------------+-------------+-------+--------------------+--------------------+---------+------+----------+--------------------------+
| id | select_type | table  | type | possible_keys  | key    | key_len | ref | rows  | Extra     |
+----+-------------+-------------+-------+--------------------+--------------------+---------+------+----------+--------------------------+
| 1 | SIMPLE  | xxx_sources | index | idx_client_channel | idx_client_channel | 1032 | NULL | 20207319 | Using where; Using index |
+----+-------------+-------------+-------+--------------------+--------------------+---------+------+----------+--------------------------+
1 row in set (0.00 sec)

看一下實際執行:
 

mysql> SELECT SQL_NO_CACHE channel, COUNT(channel) AS visitors FROM xxx_sources WHERE client_id = 1301 GROUP BY client_id, channel;
+---------+----------+
| channel | visitors |
+---------+----------+
| NULL |  0 |
+---------+----------+
1 row in set (11.69 sec)

實際執行的情況非常的糟糕.傳通的想法,這個執行從索引上執行計劃上看非常完美了,好象和MySQL沒什麼關系了. 在去看一下表的設計會發現client_id也是設計成了
varchar(255).看到這裡不防可以使用下面的方法試一下:

mysql> explain SELECT SQL_NO_CACHE channel, COUNT(channel) AS visitors FROM xxx_sources WHERE client_id = '1301' GROUP BY client_id, channel;
+----+-------------+-------------+------+--------------------+--------------------+---------+-------+--------+--------------------------+
| id | select_type | table  | type | possible_keys  | key    | key_len | ref | rows | Extra     |
+----+-------------+-------------+------+--------------------+--------------------+---------+-------+--------+--------------------------+
| 1 | SIMPLE  | xxx_sources | ref | idx_client_channel | idx_client_channel | 258  | const | 457184 | Using where; Using index |
+----+-------------+-------------+------+--------------------+--------------------+---------+-------+--------+--------------------------+
1 row in set (0.00 sec)

從執行計劃上來看,差不多,但實際差多了.具體上來看key_len從1032降到了258,執行計劃變成了const基於等於的查找,行數從原來千萬級到了十萬級了.不算也能明白IO
節省了很多.
再來看實際執行:

 

mysql> SELECT SQL_NO_CACHE channel, COUNT(channel) AS visitors FROM xxx_sources WHERE client_id = '1301' GROUP BY client_id, channel;
+---------+----------+
| channel | visitors |
+---------+----------+
| NULL |  0 |
+---------+----------+
1 row in set (0.25 sec)

哇,從11.69秒變成了0.25秒,這是什麼概念,優化了多少倍,算一下吧.

看到這裡在想什麼呢,記住這個案例,嗯,不錯,以後還可以加引號優化一下.那為什麼不問一下,能不能在優化了,為什麼會這樣呢?
我們先來看一下第一個問題:
能不能在優化了?
答案是當然可以了.從索引的長度上來看258還是一個非常大的數據,對於client_id這個字段從名字上來看,也只會存數據型的值,那為什麼不用的一個int unsigned去存呢,
索引的長度馬上會從258降到4。這樣不是又節省了很多嗎?
接下來看一下第二個問題,為什麼會這樣呢?
原因有兩點,同時基於一個原則,基於成本的優化器。對於client_id在表的定義時定義成了字符型的值,在查詢時傳入了數值型的值,需要經過一個數值轉換,悲劇的開始,最終
導致MySQL選擇了一個完成的索引去掃描。

從這個案例上,我們需要注意什麼呢?
合理的選擇數據類型,基本工太重要了,就這叫贏在起跑線,一切都不能隨便了,別把一個表定義成了降了主建外其它全是Varchar(255)。對數據庫的double/float這種字段做索引時一定要小心。

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