MySQL中的索引

来源:https://www.cnblogs.com/duanrantao/archive/2018/07/18/9331004.html
-Advertisement-
Play Games

轉載自:https://www.cnblogs.com/whgk/p/6179612.html 序言 之前寫到MySQL對錶的增刪改查(查詢最為重要)後,就感覺MySQL就差不多學完了,沒有想繼續學下去的心態了,原因可能是由於別人的影響,覺得對於MySQL來說,知道了一些複雜的查詢,就夠了,但是我認 ...


轉載自:https://www.cnblogs.com/whgk/p/6179612.html

序言

         之前寫到MySQL對錶的增刪改查(查詢最為重要)後,就感覺MySQL就差不多學完了,沒有想繼續學下去的心態了,原因可能是由於別人的影響,覺得對於MySQL來說,知道了一些複雜的查詢,就夠了,但是我認為,不管有沒有用,現在學著不懂的東西,說明就是自己薄弱的地方,多學才能比別人更強

                                        --WH

 

一、什麼是索引?為什麼要建立索引?

       索引用於快速找出在某個列中有一特定值的行,不使用索引,MySQL必須從第一條記錄開始讀完整個表,直到找出相關的行,表越大,查詢數據所花費的時間就越多,如果表中查詢的列有一個索引,MySQL能夠快速到達一個位置去搜索數據文件,而不必查看所有數據,那麼將會節省很大一部分時間。

       例如:有一張person表,其中有2W條記錄,記錄著2W個人的信息。有一個Phone的欄位記錄每個人的電話號碼,現在想要查詢出電話號碼為xxxx的人的信息。

          如果沒有索引,那麼將從表中第一條記錄一條條往下遍歷,直到找到該條信息為止。

          如果有了索引,那麼會將該Phone欄位,通過一定的方法進行存儲,好讓查詢該欄位上的信息時,能夠快速找到對應的數據,而不必在遍歷2W條數據了。其中MySQL中的索引的存儲類型有兩種:BTREE、HASH。 也就是用樹或者Hash值來存儲該欄位,要知道其中詳細是如何查找的,就需要會演算法的知識了。我們現在只需要知道索引的作用,功能是什麼就行。

 

 

二、MySQL中索引的優點和缺點和使用原則

      優點:

       2、所有的MySql列類型(欄位類型)都可以被索引,也就是可以給任意欄位設置索引

       3、大大加快數據的查詢速度

      缺點:

       1、創建索引和維護索引要耗費時間,並且隨著數據量的增加所耗費的時間也會增加

       2、索引也需要占空間,我們知道數據表中的數據也會有最大上線設置的,如果我們有大量的索引,索引文件可能會比數據文件更快達到上線值

       3、當對錶中的數據進行增加、刪除、修改時,索引也需要動態的維護,降低了數據的維護速度。

      使用原則:

            通過上面說的優點和缺點,我們應該可以知道,並不是每個欄位度設置索引就好,也不是索引越多越好,而是需要自己合理的使用。

       1、對經常更新的表就避免對其進行過多的索引,對經常用於查詢的欄位應該創建索引,

       2、數據量小的表最好不要使用索引,因為由於數據較少,可能查詢全部數據花費的時間比遍歷索引的時間還要短,索引就可能不會產生優化效果。

       3、在一同值少的列上(欄位上)不要建立索引,比如在學生表的"性別"欄位上只有男,女兩個不同值。相反的,在一個欄位上不同值較多可是建立索引。

  

      上面說的只是很片面的一些東西,索引肯定還有很多別的優點或者缺點,還有使用原則,先基本上理解索引,然後等以後真正用到了,就會慢慢知道別的作用。註意,學習這張,很重要的一點就是必須先得知道索引是什麼,索引是幹嘛的,有什麼作用,為什麼要索引等等,如果不知道,就重覆往上面看看寫的文字,好好理解一下。一個表中很夠創建多個索引,這些索引度會被存放到一個索引文件中(專門存放索引的地方)

 

 

三、索引的分類  

       註意:索引是在存儲引擎中實現的,也就是說不同的存儲引擎,會使用不同的索引

            MyISAM和InnoDB存儲引擎:只支持BTREE索引, 也就是說預設使用BTREE,不能夠更換

            MEMORY/HEAP存儲引擎:支持HASH和BTREE索引

       1、索引我們分為四類來講 單列索引(普通索引,唯一索引,主鍵索引)、組合索引、全文索引、空間索引、

          1.1、單列索引:一個索引只包含單個列,但一個表中可以有多個單列索引。 這裡不要搞混淆了。

             1.1.1、普通索引:

                  MySQL中基本索引類型,沒有什麼限制,允許在定義索引的列中插入重覆值和空值,純粹為了查詢數據更快一點。

             1.1.2、唯一索引:

                  索引列中的值必須是唯一的,但是允許為空值,

             1.1.3、主鍵索引:

                  是一種特殊的唯一索引,不允許有空值。

          1.2、組合索引

               在表中的多個欄位組合上創建的索引,只有在查詢條件中使用了這些欄位的左邊欄位時,索引才會被使用,使用組合索引時遵循最左首碼集合。這個如果還不明白,等後面舉例講解時在細說 

          1.3、全文索引

               全文索引,只有在MyISAM引擎上才能使用,只能在CHAR,VARCHAR,TEXT類型欄位上使用全文索引,介紹了要求,說說什麼是全文索引,就是在一堆文字中,通過其中的某個關鍵字等,就能找到該欄位所屬的記錄行,比如有"你是個大煞筆,二貨 ..." 通過大煞筆,可能就可以找到該條記錄。這裡說的是可能,因為全文索引的使用涉及了很多細節,我們只需要知道這個大概意思,如果感興趣進一步深入使用它,那麼看下麵測試該索引時,會給出一個博文,供大家參考。

          1.4、空間索引

               空間索引是對空間數據類型的欄位建立的索引,MySQL中的空間數據類型有四種,GEOMETRY、POINT、LINESTRING、POLYGON。

               在創建空間索引時,使用SPATIAL關鍵字。

               要求,引擎為MyISAM,創建空間索引的列,必須將其聲明為NOT NULL。具體細節看下麵   

 

四、索引操作(創建和刪除)

      4.1、創建索引

          4.1.1、創建表的時候創建索引

            格式:CREATE TABLE 表名[欄位名 數據類型]  [UNIQUE|FULLTEXT|SPATIAL|...] [INDEX|KEY] [索引名字] (欄位名[length])   [ASC|DESC]

               |--------------------------------------|  |-----------------------------------| |------------| |---------| |---------------|    |------------|

                    普通創建表語句        設置什麼樣的索引(唯一、全文等)  索引關鍵字  索引名字 對哪個欄位設置索引  對索引進行排序 

              4.1.1.1、創建普通索引              

                CREATE TABLE book                    CREATE TABLE book

                (                              (

                  bookid INT NOT NULL,                  bookid INT NOT NULL,

                  bookname VARCHAR(255) NOT NULL,           bookname VARCHAR(255) NOT NULL,

                  authors VARCHAR(255) NOT NULL,             authors VARCHAR(255) NOT NULL,

                  info VARCHAR(255) NULL,                info VARCHAR(255) NULL,

                  comment VARCHAR(255) NULL,             comment VARCHAR(255) NULL, 

                  year_publication YEAR NOT NULL,            year_publication YEAR NOT NULL,

                  INDEX(year_publication)                 KEY(year_publication) 

                );                              );

              上面兩種方式創建度可以,通過這個例子可以對比一下格式,就差不多明白格式是什麼意思了。

                        

                通過列印結果,我們在創建索引時沒寫索引名的話,會自動幫我們用欄位名當作索引名。

                測試:看是否使用了索引進行查詢。

                  EXPLAIN SELECT * FROM book WHERE year_publication = 1990\G;

                  解釋:雖然表中沒數據,但是有EXPLAIN關鍵字,用來查看索引是否正在被使用,並且輸出其使用的索引的信息。

                       

                    id: SELECT識別符。這是SELECT的查詢序列號,也就是一條語句中,該select是第幾次出現。在次語句中,select就只有一個,所以是1.

                    select_type:所使用的SELECT查詢類型,SIMPLE表示為簡單的SELECT,不實用UNION或子查詢,就為簡單的SELECT。也就是說在該SELECT查詢時會使用索引。其他取值,PRIMARY:最外面的SELECT.在擁有子查詢時,就會出現兩個以上的SELECT。UNION:union(兩張表連接)中的第二個或後面的select語句  SUBQUERY:在子查詢中,第二SELECT。

                    table:數據表的名字。他們按被讀取的先後順序排列,這裡因為只查詢一張表,所以只顯示book

                    type:指定本數據表和其他數據表之間的關聯關係,該表中所有符合檢索值的記錄都會被取出來和從上一個表中取出來的記錄作聯合。ref用於連接程式使用鍵的最左首碼或者是該鍵不是 primary key 或 unique索引(換句話說,就是連接程式無法根據鍵值只取得一條記錄)的情況。當根據鍵值只查詢到少數幾條匹配的記錄時,這就是一個不錯的連接類型。(註意,個人這裡不是很理解,百度了很多資料,全是大白話,等以後用到了這類信息時,在回過頭來補充,這裡不懂對後面的影響不大。)可能的取值有 system、const、eq_ref、index和All

                    possible_keys:MySQL在搜索數據記錄時可以選用的各個索引,該表中就只有一個索引,year_publication

                    key:實際選用的索引

                    key_len:顯示了mysql使用索引的長度(也就是使用的索引個數),當 key 欄位的值為 null時,索引的長度就是 null。註意,key_len的值可以告訴你在聯合索引中mysql會真正使用了哪些索引。這裡就使用了1個索引,所以為1,

                    ref:給出關聯關係中另一個數據表中數據列的名字。常量(const),這裡使用的是1990,就是常量。

                    rows:MySQL在執行這個查詢時預計會從這個數據表裡讀出的數據行的個數。

                    extra:提供了與關聯操作有關的信息,沒有則什麼都不寫。   

                  上面的一大堆東西能看懂多少看多少,我們最主要的是看possible_keys和key 這兩個屬性,上面顯示了key為year_publication。說明使用了索引。

             4.1.1.2、創建唯一索引  

                CREATE TABLE t1

                (

                  id INT NOT NULL,

                  name CHAR(30) NOT NULL,

                  UNIQUE INDEX UniqIdx(id)

                );  

                解釋:對id欄位使用了索引,並且索引名字為UniqIdx。

                SHOW CREATE TABLE t1\G;

                      

                  要查看其中查詢時使用的索引,必須先往表中插入數據,然後在查詢數據,不然查找一個沒有的id值,是不會使用索引的。

                INSERT INTO t1 VALUES(1,'xxx');

                EXPLAIN SELECT * FROM t1 WHERE id = 1\G;

                      

                可以看到,通過id查詢時,會使用唯一索引。並且還實驗了查詢一個沒有的id值,則不會使用索引,我覺得原因是所有的id應該會存儲到一個const tables中,到其中並沒有該id值,那麼就沒有查找的必要了。   

 

            4.1.1.3、創建主鍵索引

                CREATE TABLE t2

                (

                  id INT NOT NULL,

                  name CHAR(10),

                  PRIMARY KEY(id)

                );  

                INSERT INTO t2 VALUES(1,'QQQ');

                EXPLAIN SELECT * FROM t2 WHERE id = 1\G;

                      

                 通過這個主鍵索引,我們就應該反應過來,其實我們以前聲明的主鍵約束,就是一個主鍵索引,只是之前我們沒學過,不知道而已。

             4.1.1.4、創建單列索引 

                 這個其實就不用在說了,前面幾個就是單列索引。

 

             4.1.1.5、創建組合索引

                組合索引就是在多個欄位上創建一個索引

                創建一個表t3,在表中的id、name和age欄位上建立組合索引

                CREATE TABLE t3

                (

                  id INT NOT NULL,

                  name CHAR(30) NOT NULL,

                  age INT NOT NULL,

                  info VARCHAR(255),

                  INDEX MultiIdx(id,name,age)

                );

                SHOW CREATE t3\G;

                      

                解釋最左首碼

                      組合索引就是遵從了最左首碼,利用索引中最左邊的列集來匹配行,這樣的列集稱為最左首碼,不明白沒關係,舉幾個例子就明白了,例如,這裡由id、name和age3個欄位構成的索引,索引行中就按id/name/age的順序存放,索引可以索引下麵欄位組合(id,name,age)、(id,name)或者(id)。如果要查詢的欄位不構成索引最左面的首碼,那麼就不會是用索引,比如,age或者(name,age)組合就不會使用索引查詢

                在t3表中,查詢id和name欄位

                EXPLAIN SELECT * FROM t3 WHERE id = 1 AND name = 'joe'\G;

                      

                在t3表中,查詢(age,name)欄位,這樣就不會使用索引查詢。來看看結果

                EXPLAIN SELECT * FROM t3 WHERE age = 3 AND name = 'bob'\G;

                      

            4.1.1.6、創建全文索引

                全文索引可以用於全文搜索,但只有MyISAM存儲引擎支持FULLTEXT索引,並且只為CHAR、VARCHAR和TEXT列服務。索引總是對整個列進行,不支持首碼索引,

                CREATE TABLE t4

                (

                  id  INT NOT NULL,

                  name CHAR(30) NOT NULL,

                  age INT NOT NULL,

                  info VARCHAR(255),

                  FULLTEXT INDEX FullTxtIdx(info)

                )ENGINE=MyISAM;

                SHOW CREATE TABLE t4\G;

                    

               使用一下什麼叫做全文搜索。就是在很多文字中,通過關鍵字就能夠找到該記錄。

                  INSERT INTO t4 VALUES(8,'AAA',3,'text is so good,hei,my name is bob'),(9,'BBB',4,'my name is gorlr');

                  SELECT * FROM t4 WHERE MATCH(info) AGAINST('gorlr');

                    

                  EXPLAIN SELECT * FROM t4 WHERE MATCH(info) AGAINST('gorlr');

                    

                  註意:在使用全文搜索時,需要藉助MATCH函數,並且其全文搜索的限制比較多,比如只能通過MyISAM引擎,比如只能在CHAR,VARCHAR,TEXT上設置全文索引。比如搜索的關鍵字預設至少要4個字元,比如搜索的關鍵字太短就會被忽略掉。等等,如果你們在實驗的時候可能會實驗不出來。感興趣的同學可以看看這篇文章,全文搜索的使用

 

            4.1.1.7、創建空間索引

                空間索引也必須使用MyISAM引擎, 並且空間類型的欄位必須為非空。 這個空間索引具體能幹嘛我也不知道,可能跟游戲開發有關,可能跟別的東西有關,等遇到了自然就知道了,現在只要求能夠創建出來。

                CREATE TABLE t5

                (

                  g GEOMETRY NOT NULL,

                  SPATIAL INDEX spatIdx(g)

                ) ENGINE = MyISAM;

                SHOW CREATE TABLE t5\G;

                    

          4.1.2、在已經存在的表上創建索引

              格式:ALTER TABLE 表名 ADD[UNIQUE|FULLTEXT|SPATIAL] [INDEX|KEY] [索引名] (索引欄位名)[ASC|DESC]

                 有了上面的基礎,這裡就不用過多陳述了。

              命令一:SHOW INDEX FROM 表名\G  

                  查看一張表中所創建的索引

                  SHOW INDEX FROM book\G;

                      

                  挑重點講,我們需要瞭解的就5個,用紅顏色標記了的,如果想深入瞭解,可以去查查該方面的資料,我個人覺得,這些等以後實際工作中遇到了在做詳細的瞭解把。

                  Table:創建索引的表

                  Non_unique:表示索引非唯一,1代表 非唯一索引, 0代表 唯一索引,意思就是該索引是不是唯一索引

                  Key_name:索引名稱

                  Seq_in_index 表示該欄位在索引中的位置,單列索引的話該值為1,組合索引為每個欄位在索引定義中的順序(這個只需要知道單列索引該值就為1,組合索引為別的)

                  Column_name:表示定義索引的列欄位

                  Sub_part:表示索引的長度

                  Null:表示該欄位是否能為空值

                  Index_type:表示索引類型

             4.1.2.1、為表添加索引

                就拿上面的book表來說。本來已經有了一個year_publication,現在我們為該表在加一個普通索引

                ALTER TABLE book ADD INDEX BkNameIdx(bookname(30));

                    

                看輸出結果,就能知道,添加索引成功了。

                這裡只是拿普通索引做個例子,添加其他索引也是一樣的。依葫蘆畫瓢而已。這裡就不一一做講解了。

             4.1.2.2、使用CREATE INDEX創建索引。

                格式:CREATE [UNIQUE|FULLTEXT|SPATIAL] [INDEX|KEY] 索引名稱 ON 表名(創建索引的欄位名[length])[ASC|DESC]

                  解釋:其實就是換湯不換藥,格式改變了一下而已,做的事情跟上面完全一樣,做一個例子。

                在為book表增加一個普通索引,欄位為authors。

                CREATE INDEX BkBookNameIdx ON book(bookname);

                    

                SHOW INDEX FROM book\G;  //查看book表中的索引

                    

                  解釋:第一條截圖沒截到,因為圖太大了,這裡只要看到有我們新加進去的索引就證明成功了。。其他索引也是一樣的創建。

      4.2、刪除索引

            前面講了對一張表中索引的添加,查詢的方法。

              添加的兩種方式

                1在創建表的同時如何創建索引,

                2在創建了表之後如何給表添加索引的兩種方式,

              查詢的方式

                SHOW INDEX FROM 表名\G;  \G只是讓輸出的格式更好看

              現在來說說如何給表刪除索引的兩種操作。

                格式一:ALTER TABLE 表名 DROP INDEX 索引名。

                  很簡單的語句,現在通過一個例子來看看,還是對book表進行操作,刪除我們剛纔為其添加的索引。

                1、刪除book表中的名稱為BkBookNameIdx的索引。

                  ALTER TABLE book DROP INDEX BkBookNameIdx;

                       

                  SHOW INDEX FROM book\G;  //在查看book表中的索引,就會發現BkBookNameIdx這個索引已經不在了

                      

 

                   格式二:DROP INDEX 索引名 ON 表名;

                       刪除book表中名為BkNameIdx的索引

                      DROP INDEX BkNameIdx ON book;

                       SHOW INDEX FROM book\G;

                        

 

五、總結

      MySQL的索引到這裡差不多就講完了,總結一下我們到目前為止應該知道哪些東西

        1、索引是幹嘛的?為什麼要有索引?

            這個很重要,需要自己理解一下,不懂就看頂部的講解

        2、索引的分類

        3、索引的操作

            給表中創建索引,添加索引,刪除索引,刪除索引


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • ElasticSearch 學習筆記 Near Realtime(NRT):近實時,先說實時就是數據創建到查詢時間在毫秒級或更少; 和實時不一樣的是近實時數據在創建到查詢最多需要n秒時間,n秒是索引刷新周期(假如設置n為5秒,若你在2秒的時候插入了一條數據,那麼你還需要等3秒才能查詢到這條數據,... ...
  • 占座 ...
  • 一、ORC File文件結構 ORC的全稱是(Optimized Row Columnar),ORC文件格式是一種Hadoop生態圈中的列式存儲格式,它的產生早在2013年初,最初產生自Apache Hive,用於降低Hadoop數據存儲空間和加速Hive查詢速度。和Parquet類似,它並不是一個 ...
  • SQL ROUND函數是對數據進行制定精度的取值。 第一個參數是取值的數據,第二個參數是精度,第三個參數是數據取值模式(四捨五入還是截斷),其中第三個參數是可選參數,預設是四捨五入模式。 從上面結果可以看出,數據並非只保留兩位小數,而是保留兩位有效小數。 從上面兩次可以看出,預設不使用第三個參數是四 ...
  • mysql資料庫是一個常用的關係型資料庫 關係型資料庫核心元素有哪些? 主鍵:特殊欄位,用來唯一標識記錄的唯一性 欄位:數據列 記錄:數據行 數據表:數據行的集合 資料庫:數據表的集合 安裝、啟動、停止、重啟mysql伺服器的命令 安裝:sudo apt-get install mysql-serv ...
  • 一、主從複製 使用非同步複製 一個伺服器可以有多個從伺服器 從伺服器也可以有自己的從伺服器 複製功能不會阻塞主伺服器 可以通過服務功能來上主伺服器免於持久化操作,由從伺服器去執行持久化操作即可。 以下是關於Redis複製功能的幾個重要方面: Redis使用非同步複製。從Redis 2.8開始,從伺服器會 ...
  • 一、數據類型 二、全局Key操作 例子 三、String(字元串) 應用場景 String類型有如下基本操作 普通鍵值對操作 計數器 詳細示例 增 刪 改 查 四、Hash(字典) 應用場景 示例 存數據 取數據 詳細示例 增 刪 改 查 五、LIST(列表) 應用場景 簡單示例 更多示例 增 刪 ...
  • 之前一直出現這個錯誤,使用的開發工具是IDEA 我感覺似乎是hadoop與windows的操作系統不太適合 於是在project創建 org.apache.hadoop.io.nativeio包,將NativeIO.java中的代碼導入,然後修改 修改之後項目可運行。 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...