真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

MySQL優(yōu)化(超完整版)(一)

一、 MySQL的優(yōu)化

前言
? MySQL數(shù)據(jù)庫的優(yōu)化模塊:
? ? - 數(shù)據(jù)庫的設(shè)計(jì)—三大范式
? ? - 數(shù)據(jù)庫的索引:唯一索引、主鍵索引、聚合索引、復(fù)合索引、默認(rèn)索引
? ? - SQL優(yōu)化
? ? - 分庫分表
? ? - 讀寫分離:提升IO性能
? ? - 存儲(chǔ)過程優(yōu)化
? ? - 對(duì)MySQL配置進(jìn)行優(yōu)化(my.ini)
? ? - 定時(shí)清理碎片

從網(wǎng)站建設(shè)到定制行業(yè)解決方案,為提供成都做網(wǎng)站、成都網(wǎng)站設(shè)計(jì)服務(wù)體系,各種行業(yè)企業(yè)客戶提供網(wǎng)站建設(shè)解決方案,助力業(yè)務(wù)快速發(fā)展。成都創(chuàng)新互聯(lián)將不斷加快創(chuàng)新步伐,提供優(yōu)質(zhì)的建站服務(wù)。

1. 數(shù)據(jù)庫的設(shè)計(jì)

(1) 什么是三大范式?

? 為了建立冗余較小、結(jié)構(gòu)合理的數(shù)據(jù)庫,設(shè)計(jì)數(shù)據(jù)庫時(shí)必須遵循一定的規(guī)則。在關(guān)系型數(shù)據(jù)庫中這種規(guī)則就稱為范式。三大范式包括:
? - 1NF:屬性的原子性,要求屬性具有原子性,不可分解。
? - 2NF:對(duì)記錄的唯一性,表中記錄是唯一的(通常通過主鍵來實(shí)現(xiàn))。
? - 3NF:是對(duì)字段冗余性的約束,要求字段沒有冗余。

(2) 1NF
-- 案例
create table `user`(
id int,
name varchar(10),
address varchar(10)
);
insert into `user` values(1,’zs’,’上海市浦東新區(qū)’);
此時(shí)這里就針對(duì)address這個(gè)字段,有了可分割性,可將將address分為:-市-區(qū)。
此時(shí)表的設(shè)計(jì)就不遵循1NF。
(3) 2NF

-- 案例

create table `emp`(
id int,
name varchar(10),
age int(10),
salary float(8,4)
這里id是員工的編號(hào),每一個(gè)編號(hào)唯一確定一個(gè)員工,員工的充值和工位號(hào)也是通過這個(gè)id確定,此時(shí)用id來作為emp表的主鍵就不禁合理,因?yàn)橹麈I一般是不做業(yè)務(wù)操作的,主鍵的作用就是唯一標(biāo)識(shí)一行。
);
(4) 3NF
-- 案例
create table student(
stu_id int(10),
stu_name varchar(30),
class_id int(10),
class_name varchar(30)
);
Insert into student values(1,’zs’,1,’一班’);
Insert into student values(2,’ls’,1,’一班’);
Insert into student values(3,’ww’,1,’一班’);
Insert into student values(4,’wb’,1,’一班’);
這里我們發(fā)現(xiàn),class_id和class_name字段大量的冗余,不遵循第3NF,這里我們需要將這張表拆分:student 表和 class表
create table student(
stu_id int(10),
class_id int(10),
stu_name varchar(30)
);
create table `class`(
class_id int(10),
class_name varchar(30)
);
然后對(duì)兩張表設(shè)置外鍵關(guān)聯(lián)。

2. 慢查詢介紹及定位

(1) 定位慢查詢

? 慢查詢的定義:MySQL規(guī)定,只要10s內(nèi),沒有按照規(guī)則的時(shí)間返回結(jié)果,就是慢查詢類型,然后MySQL會(huì)將這些語句存儲(chǔ)到慢查詢?nèi)罩局小?br/>可以通過命令查看

-- 使用show status查看MySQL服務(wù)器狀態(tài)信息
mysql>show status

MySQL優(yōu)化(超完整版)(一)

-- mysql 運(yùn)行了多長時(shí)間
show status like 'uptime';

MySQL優(yōu)化(超完整版)(一)

-- 當(dāng)前窗口 數(shù)據(jù)庫的查詢次數(shù)
show status like 'com_select';

MySQL優(yōu)化(超完整版)(一)

-- 當(dāng)前窗口 數(shù)據(jù)庫的插入次數(shù)
show  status like 'com_insert';

-- 當(dāng)前窗口 數(shù)據(jù)庫的更新次數(shù)
show  status like 'com_update';

-- 當(dāng)前窗口,數(shù)據(jù)庫的刪除次數(shù)
show  status like 'com_delete';

-- 查看試圖連接到MySQL(不管是否連接成功)的連接數(shù)
show status like 'connections';

MySQL優(yōu)化(超完整版)(一)

-- 查看當(dāng)前打開的連接的數(shù)量。
show status like 'threads_connected';

MySQL優(yōu)化(超完整版)(一)

-- 顯示慢查詢數(shù)量 **********************
show status like 'slow_queries';

注意:這里默認(rèn)的是session,表示的時(shí)當(dāng)前會(huì)話,如果想查詢?nèi)值男枰?/p>

show global status like ‘’;
(2) 慢查詢案例演示

①慢查詢時(shí)間設(shè)置

--查詢慢查詢時(shí)間
show variables like 'long_query_time';
--修改慢查詢時(shí)間(臨時(shí))
set long_query_time=1; ---但是重啟mysql之后,long_query_time依然是my.ini中的值。

②建表、造數(shù)據(jù)

/*部門表*/
create
 table
  dept(
   deptno mediumint unsigned not null default 0,
   /*編號(hào)*/
   dname varchar(20) not null default "",
   /*名稱*/
   loc varchar(13) not null default "" /*地點(diǎn)*/
  ) ENGINE = MyISAM default CHARSET = utf8;

/*員工表*/
create
 table
  emp(
   empno mediumint unsigned not null default 0,
   /*編號(hào)*/
   ename varchar(20) not null default "",
   /*名字*/
   job varchar(9) not null default "",
   /*工作*/
   mgr mediumint unsigned not null default 0,
   /*上級(jí)編號(hào)*/
   hiredate date not null,
   /*入職時(shí)間*/
   sal decimal(
    7,
    2
   ) not null,
   /*薪水*/
   comm decimal(
    7,
    2
   ) not null,
   /*紅利*/
   deptno mediumint unsigned not null default 0 /*部門編號(hào)*/
  ) ENGINE = MyISAM default CHARSET = utf8;
/*薪水*/
create
 table
  salgrade(
   grade mediumint unsigned not null default 0,
   losal decimal(
    17,
    2
   ) not null,
   hisal decimal(
    17,
    2
   ) not null
  ) ENGINE = MyISAM default CHARSET = utf8;

--插入數(shù)據(jù)
INSERT INTO salgrade VALUES(1,700,1200);
INSERT INTO salgrade VALUES(2,1201,1400);
INSERT INTO salgrade VALUES(3,1401,2000);
INSERT INTO salgrade VALUES(4,2001,3000);
INSERT INTO salgrade VALUES(5,3001,9999);

③批量生成100W條數(shù)據(jù)

--生成隨機(jī)字符
create
 function rand_string(
  n int
 ) returns varchar(255) #該函數(shù)會(huì)返回一個(gè)字符串
 begin #chars_str定義一個(gè)變量 chars_str,類型是 varchar(100),默認(rèn)值'abcdefghijklmnopqrstuvwxyzABCDEFJHIJKLMNOPQRSTUVWXYZ';
 declare chars_str varchar(100) default 'abcdefghijklmnopqrstuvwxyzABCDEFJHIJKLMNOPQRSTUVWXYZ';
declare return_str varchar(255) default '';
declare i int default 0;
while i < n do
set
return_str = concat( return_str, substring( chars_str, floor( 1 + rand()* 52 ), 1 ));
set
i = i + 1;
end while;
return return_str;
end

--生成隨機(jī)數(shù)
create FUNCTION rand_num()
RETURNS int(5)
BEGIN
 DECLARE i int default 0;
 set i =floor(10+RAND()*500);
 return i;
END

--編寫存儲(chǔ)過程,插入數(shù)據(jù)
delimiter // 
create procedure insert_emp(in start int(10),in max_num int(10))
begin
declare i int default 0; 
#set autocommit =0 把a(bǔ)utocommit設(shè)置成0
 set autocommit = 0;  
 repeat
 set i = i + 1;
 insert into emp values ((start+i) ,rand_string(6),'SALESMAN',0001,curdate(),2000,400,rand_num());
  until i = max_num
 end repeat;
   commit;
 end // 
 delimiter ;

-- 執(zhí)行
call insert_emp (100001,10000000);

④設(shè)置MySQL記錄慢查詢?nèi)罩?/strong>
首先將MySQL服務(wù)關(guān)閉:
MySQL優(yōu)化(超完整版)(一)
進(jìn)入mysql/bin下執(zhí)行一下命令:
[mysql5.5 可以在my.ini指定](安全模式啟動(dòng),數(shù)據(jù)庫將操作寫入日志,以備恢復(fù))

$ mysqld.exe --safe-mode  --slow-query-log 

然后在my.ini配置文件中有這么一行:
MySQL優(yōu)化(超完整版)(一)
在這個(gè)目錄下,會(huì)生成相應(yīng)的慢查詢記錄。
#5.7版本自動(dòng)開啟:
MySQL優(yōu)化(超完整版)(一)
然后我們?cè)O(shè)置慢查詢時(shí)間為:1s

set long_query_time=1;

用剛剛造出來的數(shù)據(jù)執(zhí)行一個(gè)慢查詢:

select * from emp where ename = 'aDNehz';

查看慢查詢?nèi)罩荆?br/>MySQL優(yōu)化(超完整版)(一)
通過日志,我們就能定位到具體的是哪一條語句查詢慢。

3. MySQL的索引

索引的概述

?索引用來快速的查詢那些具有特定值的記錄。所有的MySQL索引都是以B+樹的形式保存的。如果沒有索引,執(zhí)行查詢時(shí)MySQL必須從第一個(gè)記錄開始,進(jìn)行全表掃描,直至找到合適的記錄。表里的記錄越多,這個(gè)操作越耗時(shí)。如果作為搜索條件的列上已經(jīng)創(chuàng)建了索引,MySQL無需掃描任何記錄即可迅速得到目標(biāo)記錄所在的位置。如果表有1000個(gè)記錄,通過索引查找記錄至少要比順序掃描記錄快100倍。

(1) 主鍵索引

?主鍵索引是一種唯一性索引,但是它必須指定“PRIMARY KEY”。主鍵一般在創(chuàng)建表的時(shí)候指定,并且一張表只能有一個(gè)主鍵。

#創(chuàng)建主鍵索引:
默認(rèn)情況下,MySQL會(huì)為主鍵自動(dòng)添加主鍵索引。
也可以后期添加主鍵:
Alter table table_name add primary key(field_name);
#刪除主鍵:
Alter table table_name drop primary key; 

#查看索引
show index from table_name;
show keys from table_name;
(2) 全文索引

? 全文索引一般用于查詢文本或者長內(nèi)容而建立。

CREATE TABLE articles (
       id INT UNSIGNED AUTO_INCREMENT NOT NULL PRIMARY KEY,
       title VARCHAR(200),
       body TEXT,
       FULLTEXT (title,body)
     )engine=myisam charset utf8;

INSERT INTO articles (title,body) VALUES
     ('MySQL Tutorial','DBMS stands for DataBase ...'),
     ('How To Use MySQL Well','After you went through a ...'),
     ('Optimizing MySQL','In this tutorial we will show ...'),
     ('1001 MySQL Tricks','1. Never run mysqld as root. 2. ...'),
     ('MySQL vs. YourSQL','In the following database comparison ...'),
     ('MySQL Security','When configured properly, MySQL ...');

#全文索引的使用方法
-- 錯(cuò)誤使用方法
select * from articles where body like '%mysql%';

-- 建立全文索引
alter table articles add  FULLTEXT index fulltext_article(body);
-- 正確使用方法
select * from articles where match (body) against('mysql');

使用全文索引的注意事項(xiàng):
? - MySql自帶的全文索引只能用于數(shù)據(jù)庫引擎為MYISAM的數(shù)據(jù)表,如果是其他數(shù)據(jù)引擎,則全文索引不會(huì)生效.
? - MySQL不能對(duì)中文進(jìn)行全文索引,只適用于英文.
? - 使用全文索引,只能用固定的語法:match(字段名)… against(關(guān)鍵字).
? - MySQL全文索引所能找到的默認(rèn)最小長度為4個(gè)字符,并且如果查詢的字符串包含停止詞(常見字符),那么該停止詞將會(huì)被忽略。

(3) 唯一索引

?這種索引的所有值都只能出現(xiàn)一次,即必須唯一。
默認(rèn)的,在創(chuàng)建表時(shí)指定字段為唯一時(shí),自動(dòng)為其創(chuàng)建唯一索引。

#創(chuàng)建唯一索引:
CREATE UNIQUE INDEX <索引的名字> ON tablename (列的列表);
ALTER TABLE tablename ADD UNIQUE [索引的名字] (列的列表);
CREATE TABLE tablename ( [...], UNIQUE [索引的名字] (列的列表) );
例:
create table ddd(id int primary key auto_increment , name varchar(32) unique);
create unique index uniq_index_name on ddd(name);
alter table ddd add unique uniq_index_name (name);

注意:unique字段可以為null,可以有多個(gè)null,但是如果是字符串的話只能有一個(gè)為‘’。
(4) 普通索引

? 普通索引(由關(guān)鍵字KEY或INDEX定義的索引)的唯一任務(wù)是加快對(duì)數(shù)據(jù)的訪問速度。因此,應(yīng)該只為那些最經(jīng)常出現(xiàn)在查詢條件(WHEREcolumn=)或排序條件(ORDERBYcolumn)中的數(shù)據(jù)列創(chuàng)建索引。只要有可能,就應(yīng)該選擇一個(gè)數(shù)據(jù)最整齊、最緊湊的數(shù)據(jù)列(如一個(gè)整數(shù)類型的數(shù)據(jù)列)來創(chuàng)建索引。

創(chuàng)建方式:   create index 索引名 on 表 (列1,列名2);

-- 案例
-- 未建立索引時(shí)查詢 1.45s
select * from emp where ename ='MFPkFv';
-- 創(chuàng)建普通索引
create index index_ename on emp (ename);
-- 查詢 4ms
select * from emp where ename ='MFPkFv';
(5) 索引實(shí)現(xiàn)原理

?MySQL數(shù)據(jù)庫的索引,是數(shù)據(jù)庫管理中的一個(gè)排序的數(shù)據(jù)結(jié)構(gòu),以便于協(xié)助快速查詢,更數(shù)據(jù)庫表中數(shù)據(jù)。MySQL中的索引實(shí)現(xiàn)方式就是:B+樹索引。
MySQL優(yōu)化(超完整版)(一)
?上圖展示了一種可能的索引方式。左邊是數(shù)據(jù)表,一共有兩列七條記錄,最左邊的是數(shù)據(jù)記錄的物理地址(注意邏輯上相鄰的記錄在磁盤上也并不是一定物理相鄰的)。為了加快 Col2 的查找,可以維護(hù)一個(gè)右邊所示的二叉查找樹,每個(gè)節(jié)點(diǎn)分別包含索引鍵值和一個(gè)指向?qū)?yīng)數(shù)據(jù)記錄物理地址的指針,這樣就可以運(yùn)用二叉查找在 O(log2n)的復(fù)雜度內(nèi)獲取到相應(yīng)數(shù)據(jù)。
不同的存儲(chǔ)引擎使用的索引:
MySQL優(yōu)化(超完整版)(一)
? b-/+樹索引的性能分析:先從 B-Tree 分析,根據(jù) B-Tree 的定義,可知檢索一次最多需要訪問 h 個(gè)節(jié)點(diǎn)。數(shù)據(jù)庫系統(tǒng)的設(shè)計(jì)者巧妙利用了磁盤預(yù)讀原理,將一個(gè)節(jié)點(diǎn)的大小設(shè)為等于一個(gè)頁,這樣每個(gè)節(jié)點(diǎn)只需要一次 I/O 就可以完全載入。為了達(dá)到這個(gè)目的,在實(shí)際實(shí)現(xiàn) B-Tree 還需要使用如下技巧:
?每次新建節(jié)點(diǎn)時(shí),直接申請(qǐng)一個(gè)頁的空間,這樣就保證一個(gè)節(jié)點(diǎn)物理上也存儲(chǔ)在一個(gè)頁里,加之計(jì)算機(jī)存儲(chǔ)分配都是按頁對(duì)齊的,就實(shí)現(xiàn)了一個(gè) node 只需一次 I/O。B-Tree 中一次檢索最多需要 h-1 次 I/O(根節(jié)點(diǎn)常駐內(nèi)存),漸進(jìn)復(fù)雜度為 O(h)=O(logdN)。一般實(shí)際應(yīng)用中,出度 d 是非常大的數(shù)字,通常超過 100,因此 h 非常小(通常不超過 3)。
?而紅黑樹這種結(jié)構(gòu),h 明顯要深的多。由于邏輯上很近的節(jié)點(diǎn)(父子)物理上可能很遠(yuǎn),無法利用局部性,所以紅黑樹的 I/O 漸進(jìn)復(fù)雜度也為 O(h),效率明顯比 B-Tree 差很多。
?綜上所述,用 B-Tree 作為索引結(jié)構(gòu)效率是非常高的。
? B+ 樹非葉節(jié)點(diǎn)中存放的關(guān)鍵碼并不指示數(shù)據(jù)對(duì)象的地址指針,非也節(jié)點(diǎn)只是索引部分。所有的葉節(jié)點(diǎn)在同一層上,包含了全部關(guān)鍵碼和相應(yīng)數(shù)據(jù)對(duì)象的存放地址指針,且葉節(jié)點(diǎn)按關(guān)鍵碼從小到大順序鏈接。如果實(shí)際數(shù)據(jù)對(duì)象按加入的順序存儲(chǔ)而不是按關(guān)鍵碼次數(shù)存儲(chǔ)的話,葉節(jié)點(diǎn)的索引必須是稠密索引,若實(shí)際數(shù)據(jù)存儲(chǔ)按關(guān)鍵碼次序存放的話,葉節(jié)點(diǎn)索引時(shí)稀疏索引。
?B+ 樹有 2 個(gè)頭指針,一個(gè)是樹的根節(jié)點(diǎn),一個(gè)是最小關(guān)鍵碼的葉節(jié)點(diǎn)。
所以 B+ 樹有兩種搜索方法:
?一種是按葉節(jié)點(diǎn)自己拉起的鏈表順序搜索。
?一種是從根節(jié)點(diǎn)開始搜索,和 B 樹類似,不過如果非葉節(jié)點(diǎn)的關(guān)鍵碼等于給定值,搜索并不停止,而是繼續(xù)沿右指針,一直查到葉節(jié)點(diǎn)上的關(guān)鍵碼。所以無論搜索是否成功,都將走完樹的所有層。
B+ 樹中,數(shù)據(jù)對(duì)象的插入和刪除僅在葉節(jié)點(diǎn)上進(jìn)行。

綜上所述,兩種排序的不同之處在于:
? - B 樹中同一鍵值不會(huì)出現(xiàn)多次,并且它有可能出現(xiàn)在葉結(jié)點(diǎn),也有可能出現(xiàn)在非葉結(jié)點(diǎn)中。而 B+ 樹的鍵一定會(huì)出現(xiàn)在葉結(jié)點(diǎn)中,并且有可能在非葉結(jié)點(diǎn)中也有可能重復(fù)出現(xiàn),以維持 B+ 樹的平衡。
? - 因?yàn)?B 樹鍵位置不定,且在整個(gè)樹結(jié)構(gòu)中只出現(xiàn)一次,雖然可以節(jié)省存儲(chǔ)空間,但使得在插入、刪除操作復(fù)雜度明顯增加。B+ 樹相比來說是一種較好的折中。
? - B 樹的查詢效率與鍵在樹中的位置有關(guān),最大時(shí)間復(fù)雜度與 B+ 樹相同(在葉結(jié)點(diǎn)的時(shí)候),最小時(shí)間復(fù)雜度為 1(在根結(jié)點(diǎn)的時(shí)候)。而 B+ 樹的時(shí)候復(fù)雜度對(duì)某建成的樹是固定的??梢話呙?的次方。

(6) 索引的優(yōu)勢與劣勢

優(yōu)勢:
? - 創(chuàng)建索引可以大大提高系統(tǒng)性能
? - 大大加快對(duì)數(shù)據(jù)的檢索速度
? - 加速表和表之間的連接
? - 對(duì)使用分組和排序子句進(jìn)行數(shù)據(jù)檢索時(shí),同樣可以顯著減少查詢中分組和排序的時(shí)間
劣勢:
? - 建立索引耗時(shí):創(chuàng)建索引和維護(hù)索引要耗費(fèi)時(shí)間,這種時(shí)間隨著數(shù)據(jù)量的增加而增加
? - 占用一定的物理內(nèi)存:索引需要占物理空間,除了數(shù)據(jù)表占數(shù)據(jù)空間之外,每一個(gè)索引還要占一定的物理空間,如果要建立聚簇索引,那么需要的空間就會(huì)更大。
? - DML操作效率變低:當(dāng)對(duì)表中的數(shù)據(jù)進(jìn)行增加、刪除和修改的時(shí)候,索引也要?jiǎng)討B(tài)的維護(hù),這樣就降低了數(shù)據(jù)的維護(hù)速度。

(7) 建立索引的時(shí)機(jī)

應(yīng)該建立索引的字段:
? - 查詢作為查詢條件字段應(yīng)該創(chuàng)建索引
? - 經(jīng)常用在連接的字段可以建立索引
? - 經(jīng)常需要根據(jù)范圍進(jìn)行搜索的列上創(chuàng)建索引
? - 經(jīng)常需要排序的列上創(chuàng)建索引
不應(yīng)建立索引的字段:
? - 那些在查詢中很少使用或者參考的列不應(yīng)該創(chuàng)建索引
? - 對(duì)于那些只有很少數(shù)據(jù)值的列也不應(yīng)該增加索引,例如:性別、是否已婚等等
? - 對(duì)于那些定義為 text, image 和 bit 數(shù)據(jù)類型的列不應(yīng)該增加索引
? - 修改性能遠(yuǎn)遠(yuǎn)大于檢索性能時(shí),不應(yīng)該創(chuàng)建索引

(8) 索引使用的注意事項(xiàng)

① 聯(lián)合索引的失效場景

-- 給dept新增數(shù)據(jù):
create PROCEDURE insert_dept(in start int(10),in max_num int(10))
BEGIN
 declare i int DEFAULT 0;
 set autocommit=0;
 REPEAT
 set i=i+1;
 insert into dept values ((start+i),rand_string(10),rand_string(8));
 UNTIL i =max_num
 end REPEAT;
  commit;
END

-- 執(zhí)行
call insert_dept(100,10);
-- 創(chuàng)建主鍵索引
alter table dept add primary key (deptno);
-- 創(chuàng)建一個(gè)聯(lián)合索引
alter table dept add index my_ind (dname,loc); //  dname 左邊的列,loc就是右邊的列
在聯(lián)合索引中,以dept為例:
-- 索引不失效
explain select * from dept where dname ='fuHagHPcRc';
-- 索引不失效
explain select * from dept where dname ='fuHagHPcRc' and loc = 'KtPSTRAc';
-- 索引失效
explain select * from dept where  loc = 'KtPSTRAc';

綜上案例得出:創(chuàng)建的多列索引,如果不是使用第一部分,則不會(huì)創(chuàng)建索引。

② 模糊查詢時(shí)like,索引不會(huì)失效,但是如果like中有‘%xxx%’,則索引失效,但是%放在后面索引不會(huì)失效,例“xxx%”,但是%放在前面會(huì)失效,例:“%xxx”。
③ 如果條件中有or,及時(shí)其中有帶索引字段,也不會(huì)使用索引
④ 如果類型為字符串,那一定要在條件中將數(shù)據(jù)使用引號(hào)引用起來。否則不使用索引
⑤ 如果MySQL的全表掃描比使用索引快,則不使用索引。

#查看索引的使用率:
show status like 'handler_read%';

MySQL優(yōu)化(超完整版)(一)

handler_read_key:這個(gè)值越高越好,越高表示使用索引查詢到的次數(shù)。
handler_read_rnd_next:這個(gè)值越高,說明查詢低效。

4. SQL語句優(yōu)化技巧

(1) group by 的默認(rèn)排序

? 使用group by 分組查詢是,默認(rèn)分組后,還會(huì)排序,可能會(huì)降低速度,在group by 后面增加 order by null就可以防止排序。

-- 案例
-- 8.188s
select * from emp  group by deptno;

 -- 6.200s
select * from emp  group by deptno order by null;

這是因?yàn)樵趃roup by后默認(rèn)使用排序:
explain select * from emp  group by deptno;

MySQL優(yōu)化(超完整版)(一)

(2) 有些情況下,使用連接來代替子查詢

原因:使用join,MySQL不需要在內(nèi)存中創(chuàng)建臨時(shí)表。

-- 7.924s
select * from dept, emp where dept.deptno=emp.deptno;

-- 8.8s
select * from dept left join emp on dept.deptno=emp.deptno;
(3) 查詢優(yōu)化時(shí),盡量避免全表掃描
(4) 盡量避免在where 子句后面對(duì)字段進(jìn)行null值判斷,否則將導(dǎo)致引擎放棄使用索引而進(jìn)行全表掃描(解決,對(duì)null值進(jìn)行default 設(shè)置)
(5) 在對(duì)where子句進(jìn)行判斷時(shí),盡量不要使用xx=,例:>=、<=,比如>=18 就寫成>17
(6) 能用between 就不要用in

二、 補(bǔ)充內(nèi)容

1. MySQL的存儲(chǔ)引擎

(1) MySQL引擎介紹

? MySQL使用的存儲(chǔ)引擎有三種:myisam/ innodb/ memory。
? Myisam存儲(chǔ):如果表對(duì)事務(wù)要求不高,同時(shí)是以查詢和添加為主的,我們考慮使用myisam存儲(chǔ)引擎. ,比如 bbs 中的 發(fā)帖表,回復(fù)表。
? innodb存儲(chǔ):對(duì)事務(wù)要求高,保存的數(shù)據(jù)都是重要數(shù)據(jù),我們建議使用INNODB,比如訂單表,賬號(hào)表。
? Memory:基于內(nèi)存存儲(chǔ),類似于redis。

(2) myisam引擎與innodb引擎的區(qū)別

?- 事物安全(MyISAM不支持事務(wù),INNODB支持事務(wù))
?- 查詢和添加速度(MyISAM批量插入速度快)
?- 支持全文索引(MyISAM支持全文索引,INNODB不支持全文索引)
?- 鎖機(jī)制(MyISAM時(shí)表鎖,innodb是行鎖)
?- 外鍵機(jī)制(MyISAM 不支持外鍵, INNODB支持外鍵)
Ps:Memory 存儲(chǔ),比如我們數(shù)據(jù)變化頻繁,不需要入庫,同時(shí)又頻繁的查詢和修改,我們考慮使用memory, 速度極快。
MySQL優(yōu)化(超完整版)(一)

(3) myisam使用的注意事項(xiàng)

?如果我們?cè)诮ū淼臅r(shí)候指定的是myisam引擎,那么在我們刪除表數(shù)據(jù)時(shí),默認(rèn)只是邏輯刪除,而真正的物理存儲(chǔ)的數(shù)據(jù)文件是不會(huì)刪除的。
?接下來由下圖,為大家講解一下MySQL數(shù)據(jù)存放的方式:
?在my.ini文件中有:datadir=C:/ProgramData/MySQL/MySQL Server 5.7/Data這個(gè)配置就是本地系統(tǒng)存放MySQL數(shù)據(jù)文件的目錄。
MySQL優(yōu)化(超完整版)(一)
進(jìn)入具體的數(shù)據(jù)庫:(test)
MySQL優(yōu)化(超完整版)(一)
這其中:
?.frm結(jié)尾的表示表的結(jié)構(gòu)文件
?.MYD結(jié)尾的表示數(shù)據(jù)文件
?.MYI結(jié)尾的表示索引文件
#然后我們演示如果使用的是myisam引擎,數(shù)據(jù)如何刪除:

-- 建表,并指定引擎為myisam
create table test100(id int unsigned ,name varchar (22)) engine=myisam;
-- 插入數(shù)據(jù)
insert into test100 values(1,'aaaaa');
insert into test100 values(2,'bbbb');
insert into test100 values(3,'ccccc');
-- 批量增加數(shù)據(jù)
insert into test100 select id,name from test100;
-- 刪除部分?jǐn)?shù)據(jù)
delete from test100 where id =2;

此時(shí)我們查看test100.MYD 發(fā)現(xiàn)這個(gè)文件的大小根本沒有變化。

--這時(shí)我們必須清除碎片:
optimize table test100;  

之后test100.MYD大小就會(huì)改變了,數(shù)據(jù)真正意義上的刪除了。

2. MySQL的定時(shí)備份

編寫備份腳本,然后通過crontab 定時(shí)執(zhí)行。

-- 語法:
mysqldump –u -賬號(hào) –密碼 數(shù)據(jù)庫 [表名1 表名2..]  > 文件路徑
--例:
mysqldump -u -root root test > d:\temp.sql
--載入數(shù)據(jù),在MySQL
mysql>source /path/temp.sql

小編在下篇文章給大家MySQL如何分庫分表,MySQL負(fù)載均衡版的讀寫分離。

博文地址:https://blog.51cto.com/14048416/2432938

絕對(duì)沒有水貨?。。。。。。。。?/p>
分享文章:MySQL優(yōu)化(超完整版)(一)
本文地址:http://weahome.cn/article/jdedsj.html

其他資訊

在線咨詢

微信咨詢

電話咨詢

028-86922220(工作日)

18980820575(7×24)

提交需求

返回頂部