今天樓主給你們列一下關於數據庫幾個常見問題的要點,若是你們對其中的問題感興趣,能夠自行擴展研究。前端
UNION和UNION ALL關鍵字都是將兩個結果集合併爲一個。mysql
UNION在進行表連接後會篩選掉重複的記錄,因此在表連接後會對所產生的結果集進行排序運算,刪除重複的記錄再返回結果。sql
而UNION ALL只是簡單的將兩個結果合併後就返回。數據庫
因爲UNION須要排序去重,因此 UNION ALL 的效率比 UNION 好不少。segmentfault
0 rows affected
,能夠解釋爲沒有返回結果。相同點
緩存
YYYY-MM-DD HH:MM:SS
。不一樣點
性能
TIMESTAMP
1970-01-01 08:00:01~2038-01-19 11:14:07
。DATETIME
1000-10-01 00:00:00~9999-12-31 23:59:59
。兩個或更多個列上的索引被稱做聯合索引,聯合索引又叫複合索引。mysql索引
減小開銷
:建一個聯合索引(col1,col2,col3),實際至關於建了(col1),(col1,col2),(col1,col2,col3)三個索引。減小磁盤空間的開銷。覆蓋索引
:對聯合索引(col1,col2,col3),若是有以下的sql: select col1,col2,col3 from test where col1=1 and col2=2。那麼MySQL能夠直接經過遍歷索引取得數據,而無需回表,這減小了不少的隨機io操做。覆蓋索引是主要的提高性能的優化手段之一。效率高
:索引列越多,經過索引篩選出的數據越少。有1000W條數據的表,有以下sql select from table where col1=1 and col2=2 and col3=3
,假設假設每一個條件能夠篩選出10%的數據,若是隻有單值索引,那麼經過該索引能篩選出1000W*10%=100w
條數據,而後再回表從100w條數據中找到符合col2=2 and col3= 3的數據,而後再排序,再分頁;若是是聯合索引,經過索引篩選出1000w*10%*10%*10%=1w
,效率獲得明顯提高。a = 1 and b = 2 and c > 3 and d = 4
若是創建(a,b,c,d)
順序的索引,d是用不到索引的,若是創建(a,b,d,c)
的索引則均可以用到,a,b,d的順序能夠任意調整。a = 1 and b = 2 and c = 3
創建(a,b,c)
索引能夠任意順序,mysql的查詢優化器會幫你優化成索引能夠識別的形式。前綴索引就是對文本的前幾個字符(具體是幾個字符在建立索引時指定)建立索引,這樣建立起來的索引更小。可是MySQL不能在ORDER BY或GROUP BY中使用前綴索引,也不能把它們用做覆蓋索引。優化
建立前綴索引的語法:spa
ALTER TABLE table_name ADD
KEY(column_name(prefix_length))
複製代碼
簡單的說:
主索引的區別
:InnoDB的數據文件自己就是索引文件。而MyISAM的索引和數據是分開的。
輔助索引的區別
:InnoDB的輔助索引data域存儲相應記錄主鍵的值而不是地址。而MyISAM的輔助索引和主索引沒有多大區別。
InnoDB中數據記錄自己被存於主索引(B+樹)的葉子節點上。這就要求同一個葉子節點內(大小爲一個內存頁或磁盤頁)的各條數據記錄按主鍵順序存放,所以每當有一條新的記錄插入時,MySQL會根據其主鍵將其插入適當的結點和位置,若是頁面達到裝載因子(InnoDB默認爲15/16),則開闢一個新的頁。
若是使用自增主鍵,那麼每次插入新的記錄,記錄就會順序添加到當前索引結點的後續位置,當一頁寫滿,就會自動開闢一個新的頁,這樣就會造成一個緊湊的索引結構,近似順序填滿。因爲每次插入時也不須要移動已有數據,所以效率很高,也不會增長不少開銷在維護索引上。
若是使用非自增主鍵,因爲每次插入主鍵的值近似於隨機,所以每次新紀錄都要被插入到現有索引頁的中間某個位置,此時MySQL不得不爲了將新記錄查到合適位置而移動元素,甚至目標頁可能已經被回寫到磁盤上而從緩存中清掉,此時又要從磁盤上讀回來,這增長了不少開銷,同時頻繁的移動、分頁操做形成了大量的碎片,獲得了不夠緊湊的索引結構,後續不得不經過 OPTIMIZE TABLE
來重建表並優化填充頁面。
簡單的說:
索引樹只能定位到某一頁,每一頁內的插入仍是須要經過比較、移動插入的。因此有序主鍵能夠提高插入效率。
int佔多少個字節,已是固定的了,長度表明了顯示的最大寬度。若是不夠會用0在左邊填充,但必須搭配zerofill使用。也就是說,int的長度並不影響數據的存儲精度,長度只和顯示有關。
Table
:
Non_unique
:
0
:該索引不含重複值。1
:該索引可含有重複值。Key_name
:
PRIMARY
。Seq_in_index
:
idx_a_b_c (
a,
b,
c)
,則a的Seq_in_index
=1,b=2,c=3。Column_name
:
Collation
:
Cardinality
:
ANALYZE TABLE
(INNODB) 或者 myisamchk -a
(MyISAM)更新該值。Sub_part
:
NULL
。Null
:
YES
:該列容許NULL值。''
:該列不容許NULL值。Index_type
:
LIKE問題
:like 以通配符開頭 ('%abc...'),mysql索引失效會變成全表掃描的操做。
LIKE
,LIKE
條件是 type = range
級別%xxx%
:全表掃描%xxx
:全表掃描xxx%
:range解決辦法
:
ALL
變爲INDEX
,爲啥呢?覆蓋索引以後就能使用使用索引進行全表掃描。這裏要注意一下,使用符合索引的時候,命中一個字段就能夠,不用所有命中。SELECT * FROM ttl_product_info ORDER BY id LIMIT N,M
。其中 LIMIT N,M
存在的問題最大:取出N+M行,丟棄前N行,返回 N ~ N+M
行的記錄,若是N值很是大,效率極差(表記錄1500w,N=10000000,M=30 須要9秒)。SELECT id FROM ttl_product_info WHERE id > N LIMIT M
,id 列是索引列,id > N
屬於 range
級別,效率天然高,而後從位置開始取30條記錄,效率極高(表記錄1500w,N=10000000,M=30,須要0.9毫秒)。爲了保持文章結構的完整性,這裏強行加上一段總結。。。
參考文章: