1. 程式人生 > >mysql數據庫從刪庫到跑路之mysql數據類型

mysql數據庫從刪庫到跑路之mysql數據類型

sorted 指定 概覽 ember oat 取出 綜合練習 columns .html

一 介紹

存儲引擎決定了表的類型,而表內存放的數據也要有不同的類型,每種數據類型都有自己的寬度,但寬度是可選的

詳細參考:

  • http://www.runoob.com/mysql/mysql-data-types.html
  • http://dev.mysql.com/doc/refman/5.7/en/data-type-overview.html

mysql數據類型概覽

1 數字(默認都是有符號,寬度指的是顯示寬度,與存儲無關)
    tinyint int bigint:個數,年齡,id,qq號,手機號
    小數:
        float:價格,身高,體重,余額

2 字符(寬度指的是字符個數):姓名,性別,職業,地址,職稱,介紹
    char:簡單粗暴,不夠則湊夠固定長度存放起來,浪費空間,存取速度快
    varchar:精準,計算出待存放的數據的長度,節省空間,存取速度慢

3 日期時間類型: #註冊時間 datetime 2017-09-06 10:39:49 #最常用 #出生年月日,開學時間 date:2017-09-06 #聊天記錄,上課時間 time:10:39:49 #出生年 year:2017 4 枚舉與集合 enum枚舉:規定一個範圍,可有多個值,但是為該字段傳值時,只能取規定範圍中的一個 set集合:規定一個範圍,可有多個值,但是為該字段傳值時,可以取規定範圍中的一個或多個

二 數值類型

整數類型:tinyint smallint meduumint int bigint

作用:存儲年齡,等級,id,各種號碼等

技術分享
========================================
        tinyint[(m)] [unsigned] [zerofill]

            小整數,數據類型用於保存一些範圍的整數數值範圍:
            有符號:
                -128 ~ 127
            無符號:
                0 ~ 255

            PS: MySQL中無布爾值,使用tinyint(1)構造。



========================================
        int[(m)][unsigned][zerofill]

            整數,數據類型用於保存一些範圍的整數數值範圍:
            有符號:
                    
-2147483648 ~ 2147483647 無符號: 0 ~ 4294967295 ======================================== bigint[(m)][unsigned][zerofill] 大整數,數據類型用於保存一些範圍的整數數值範圍: 有符號: -9223372036854775808 ~ 9223372036854775807 無符號: 0 ~ 18446744073709551615
整數類型介紹 技術分享
=========有符號和無符號tinyint==========
#tinyint默認為有符號
mysql> create table t5(x tinyint);
Query OK, 0 rows affected (0.06 sec)

mysql> insert into t5 values
    -> (-129),
    -> (-128),
    -> (127), 
    -> (128);
Query OK, 4 rows affected, 2 warnings (0.00 sec)
Records: 4  Duplicates: 0  Warnings: 2

mysql> select * from t5;
+------+
| x    |
+------+
| -128 |        #-129存成了-128
| -128 |        #有符號,最小值為-128
|  127 |         #有符號最大值127
|  127 |         #128存成了127
+------+
4 rows in set (0.00 sec)
#設置無符號tinyint
MariaDB [db1]> create table t2(x tinyint unsigned);
MariaDB [db1]> insert into t2 values
    -> (-1),
    -> (0),
    -> (255),
    -> (256);
MariaDB [db1]> select * from t2;
+------+
| x    |
+------+
|    0 | -1存成了0
|    0 | #無符號,最小值為0
|  255 | #無符號,最大值為255
|  255 | #256存成了255
+------+
============有符號和無符號int=============
#int默認為有符號
MariaDB [db1]> create table t3(x int); #默認為有符號整數
MariaDB [db1]> insert into t3 values
    -> (-2147483649),
    -> (-2147483648),
    -> (2147483647),
    -> (2147483648);
MariaDB [db1]> select * from t3;
+-------------+
| x           |
+-------------+
| -2147483648 | #-2147483649存成了-2147483648
| -2147483648 | #有符號,最小值為-2147483648
|  2147483647 | #有符號,最大值為2147483647
|  2147483647 | #2147483648存成了2147483647
+-------------+


#設置無符號int
MariaDB [db1]> create table t4(x int unsigned);
MariaDB [db1]> insert into t4 values
    -> (-1),
    -> (0),
    -> (4294967295),
    -> (4294967296);
MariaDB [db1]> select * from t4;
+------------+
| x          |
+------------+
|          0 | #-1存成了0
|          0 | #無符號,最小值為0
| 4294967295 | #無符號,最大值為4294967295
| 4294967295 | #4294967296存成了4294967295
+------------+





==============有符號和無符號bigint=============
MariaDB [db1]> create table t6(x bigint);
MariaDB [db1]> insert into t5 values  
    -> (-9223372036854775809),
    -> (-9223372036854775808),
    -> (9223372036854775807),
    -> (9223372036854775808);

MariaDB [db1]> select * from t5;
+----------------------+
| x                    |
+----------------------+
| -9223372036854775808 |
| -9223372036854775808 |
|  9223372036854775807 |
|  9223372036854775807 |
+----------------------+



MariaDB [db1]> create table t6(x bigint unsigned);
MariaDB [db1]> insert into t6 values  
    -> (-1),
    -> (0),
    -> (18446744073709551615),
    -> (18446744073709551616);

MariaDB [db1]> select * from t6;
+----------------------+
| x                    |
+----------------------+
|                    0 |
|                    0 |
| 18446744073709551615 |
| 18446744073709551615 |
+----------------------+



======用zerofill測試整數類型的顯示寬度=============
MariaDB [db1]> create table t7(x int(3) zerofill);
MariaDB [db1]> insert into t7 values
    -> (1),
    -> (11),
    -> (111),
    -> (1111);
MariaDB [db1]> select * from t7;
+------+
| x    |
+------+
|  001 |
|  011 |
|  111 |
| 1111 | #超過寬度限制仍然可以存
+------+
整數型

!!!註意:為該類型指定寬度時,僅僅只是指定查詢結果的顯示寬度,與存儲範圍無關,存儲範圍如下

     其實我們完全沒必要為整數類型指定顯示寬度,使用默認的就可以了

     默認的顯示寬度,都是在最大值的基礎上加1

技術分享

int的存儲寬度是4個Bytes,即32個bit,即2**32

無符號最大值為:4294967296-1

有符號最大值:2147483648-1

有符號和無符號的最大數字需要的顯示寬度均為10,而針對有符號的最小值則需要11位才能顯示完全,所以int類型默認的顯示寬度為11是非常合理的

最後:整形類型,其實沒有必要指定顯示寬度,使用默認的就ok

  定點數類型 DEC等同於DECIMAL  

  浮點類型:FLOAT DOUBLE

  作用:存儲薪資、身高、體重、體質參數等

技術分享
======================================
        decimal[(m[,d])] [unsigned] [zerofill]
            準確的小數值,m是數字總個數(負號不算),d是小數點後個數。 m最大值為65,d最大值為30。

            特別的:對於精確數值計算時需要用此類型
                   decaimal能夠存儲精確值的原因在於其內部按照字符串存儲。



======================================
        FLOAT[(M,D)] [UNSIGNED] [ZEROFILL]
            單精度浮點數(非準確小數值),m是數字總個數,d是小數點後個數。
                有符號:
                    -3.402823466E+38 to -1.175494351E-38,
                    0
                    1.175494351E-38 to 3.402823466E+38
                無符號:
                    0
                    1.175494351E-38 to 3.402823466E+38

            **** 數值越大,越不準確 ****


======================================
        DOUBLE[(M,D)] [UNSIGNED] [ZEROFILL]
            雙精度浮點數(非準確小數值),m是數字總個數,d是小數點後個數。

                有符號:
                    -1.7976931348623157E+308 to -2.2250738585072014E-308
                    0
                    2.2250738585072014E-308 to 1.7976931348623157E+308
                無符號:
                    0
                    2.2250738585072014E-308 to 1.7976931348623157E+308
            **** 數值越大,越不準確 ****
View Code 技術分享
MariaDB [db1]> create table t8(salary float(5,2)); #總共5位,小數部分占2位,因而整數部分最多3位
MariaDB [db1]> insert into t8 values
    -> (3.3),
    -> (7.33),
    -> (9.335),
    -> (1000.1);
MariaDB [db1]> select * from t8;
+--------+
| salary |
+--------+
|   3.30 |
|   7.33 |
|   9.34 | #4舍5入
| 999.99 | #小數最多2位,整數最多3位
+--------+
驗證

 位類型:BIT

  BIT(M)可以用來存放多位二進制數,M範圍從1~64,如果不寫默認為1位。
  註意:對於位字段需要使用函數讀取
      bin()顯示為二進制
      hex()顯示為十六進制

MariaDB [db1]> create table t9(id bit);
MariaDB [db1]> desc t9; #bit默認寬度為1
+-------+--------+------+-----+---------+-------+
| Field | Type   | Null | Key | Default | Extra |
+-------+--------+------+-----+---------+-------+
| id    | bit(1) | YES  |     | NULL    |       |
+-------+--------+------+-----+---------+-------+

MariaDB [db1]> insert into t9 values(8);
MariaDB [db1]> select * from t9; #直接查看是無法顯示二進制位的
+------+
| id   |
+------+
|     |
+------+
MariaDB [db1]> select bin(id),hex(id) from t9; #需要轉換才能看到
+---------+---------+
| bin(id) | hex(id) |
+---------+---------+
| 1       | 1       |
+---------+---------+

MariaDB [db1]> alter table t9 modify id bit(5);
MariaDB [db1]> insert into t9 values(8);
MariaDB [db1]> select bin(id),hex(id) from t9;
+---------+---------+
| bin(id) | hex(id) |
+---------+---------+
| 1       | 1       |
| 1000    | 8       |
+---------+---------+

三 日期類型

3 日期
#註冊時間
datetime 2017-09-06 10:39:49

#出生年月日,開學時間
date:2017-09-06

#聊天記錄,上課時間
time:10:39:49

#出生年
year:2017

        YEAR
            YYYY(1901/2155)

        DATE
            YYYY-MM-DD(1000-01-01/9999-12-31)

        TIME
            HH:MM:SS(-838:59:59/838:59:59)

        DATETIME

            YYYY-MM-DD HH:MM:SS(1000-01-01 00:00:00/9999-12-31 23:59:59    Y)

        TIMESTAMP

            YYYYMMDD HHMMSS(1970-01-01 00:00:00/2037 年某時)
技術分享
============year===========
MariaDB [db1]> create table t10(born_year year); #無論year指定何種寬度,最後都默認是year(4)
MariaDB [db1]> insert into t10 values  
    -> (1900),
    -> (1901),
    -> (2155),
    -> (2156);
MariaDB [db1]> select * from t10;
+-----------+
| born_year |
+-----------+
|      0000 |
|      1901 |
|      2155 |
|      0000 |
+-----------+


============date,time,datetime===========
MariaDB [db1]> create table t11(d date,t time,dt datetime);
MariaDB [db1]> desc t11;
+-------+----------+------+-----+---------+-------+
| Field | Type     | Null | Key | Default | Extra |
+-------+----------+------+-----+---------+-------+
| d     | date     | YES  |     | NULL    |       |
| t     | time     | YES  |     | NULL    |       |
| dt    | datetime | YES  |     | NULL    |       |
+-------+----------+------+-----+---------+-------+

MariaDB [db1]> insert into t11 values(now(),now(),now());
MariaDB [db1]> select * from t11;
+------------+----------+---------------------+
| d          | t        | dt                  |
+------------+----------+---------------------+
| 2017-07-25 | 16:26:54 | 2017-07-25 16:26:54 |
+------------+----------+---------------------+



============timestamp===========
MariaDB [db1]> create table t12(time timestamp);
MariaDB [db1]> insert into t12 values();
MariaDB [db1]> insert into t12 values(null);
MariaDB [db1]> select * from t12;
+---------------------+
| time                |
+---------------------+
| 2017-07-25 16:29:17 |
| 2017-07-25 16:30:01 |
+---------------------+



============註意啦,註意啦,註意啦===========
1. 單獨插入時間時,需要以字符串的形式,按照對應的格式插入
2. 插入年份時,盡量使用4位值
3. 插入兩位年份時,<=69,以20開頭,比如50,  結果2050      
                >=70,以19開頭,比如71,結果1971
MariaDB [db1]> create table t12(y year);
MariaDB [db1]> insert into t12 values  
    -> (50),
    -> (71);
MariaDB [db1]> select * from t12;
+------+
| y    |
+------+
| 2050 |
| 1971 |
+------+



============綜合練習===========
MariaDB [db1]> create table student(
    -> id int,
    -> name varchar(20),
    -> born_year year,
    -> birth date,
    -> class_time time,
    -> reg_time datetime);

MariaDB [db1]> insert into student values
    -> (1,alex,"1995","1995-11-11","11:11:11","2017-11-11 11:11:11"),
    -> (2,egon,"1997","1997-12-12","12:12:12","2017-12-12 12:12:12"),
    -> (3,wsb,"1998","1998-01-01","13:13:13","2017-01-01 13:13:13");

MariaDB [db1]> select * from student;
+------+------+-----------+------------+------------+---------------------+
| id   | name | born_year | birth      | class_time | reg_time            |
+------+------+-----------+------------+------------+---------------------+
|    1 | alex |      1995 | 1995-11-11 | 11:11:11   | 2017-11-11 11:11:11 |
|    2 | egon |      1997 | 1997-12-12 | 12:12:12   | 2017-12-12 12:12:12 |
|    3 | wsb  |      1998 | 1998-01-01 | 13:13:13   | 2017-01-01 13:13:13 |
+------+------+-----------+------------+------------+---------------------+
驗證 技術分享
在實際應用的很多場景中,MySQL的這兩種日期類型都能夠滿足我們的需要,存儲精度都為秒,但在某些情況下,會展現出他們各自的優劣。下面就來總結一下兩種日期類型的區別。

1.DATETIME的日期範圍是1001——9999年,TIMESTAMP的時間範圍是1970——2038年。

2.DATETIME存儲時間與時區無關,TIMESTAMP存儲時間與時區有關,顯示的值也依賴於時區。在mysql服務器,操作系統以及客戶端連接都有時區的設置。

3.DATETIME使用8字節的存儲空間,TIMESTAMP的存儲空間為4字節。因此,TIMESTAMP比DATETIME的空間利用率更高。

4.DATETIME的默認值為null;TIMESTAMP的字段默認不為空(not null),默認值為當前時間(CURRENT_TIMESTAMP),如果不做特殊處理,並且update語句中沒有指定該列的更新值,則默認更新為當前時間。
datetime於timestamp的區別

四 字符串類型

#官網:https://dev.mysql.com/doc/refman/5.7/en/char.html
#註意:char和varchar括號內的參數指的都是字符的長度

#char類型:定長,簡單粗暴,浪費空間,存取速度快
    字符長度範圍:0-255(一個中文是一個字符,是utf8編碼的3個字節)
    存儲:
        存儲char類型的值時,會往右填充空格來滿足長度
        例如:指定長度為10,存>10個字符則報錯,存<10個字符則用空格填充直到湊夠10個字符存儲

    檢索:
        在檢索或者說查詢時,查出的結果會自動刪除尾部的空格,除非我們打開pad_char_to_full_length SQL模式(SET sql_mode = PAD_CHAR_TO_FULL_LENGTH;)

#varchar類型:變長,精準,節省空間,存取速度慢
    字符長度範圍:0-65535(如果大於21845會提示用其他類型 。mysql行最大限制為65535字節,字符編碼為utf-8:https://dev.mysql.com/doc/refman/5.7/en/column-count-limit.html)
    存儲:
        varchar類型存儲數據的真實內容,不會用空格填充,如果ab  ,尾部的空格也會被存起來
        強調:varchar類型會在真實數據前加1-2Bytes的前綴,該前綴用來表示真實數據的bytes字節數(1-2Bytes最大表示65535個數字,正好符合mysql對row的最大字節限制,即已經足夠使用)
        如果真實的數據<255bytes則需要1Bytes的前綴(1Bytes=8bit 2**8最大表示的數字為255)
        如果真實的數據>255bytes則需要2Bytes的前綴(2Bytes=16bit 2**16最大表示的數字為65535)
    
    檢索:
        尾部有空格會保存下來,在檢索或者說查詢時,也會正常顯示包含空格在內的內容
技術分享
#官網:https://dev.mysql.com/doc/refman/5.7/en/char.html
CHAR 和 VARCHAR 是最常使用的兩種字符串類型。
一般來說
CHAR(N)用來保存固定長度的字符串,對於 CHAR 類型,N 的範圍 為 0 ~ 255
VARCHAR(N)用來保存變長字符類型,對於 VARCHAR 類型,N 的範圍為 0 ~ 65 535
CHAR(N)和 VARCHAR(N) 中的 N 都代表字符長度,而非字節長度。
ps:對於 MySQL 4.1 之前的版本,如 MySQL 3.23 和 MySQL 4.0,CHAR(N)和 VARCHAR (N)中的 N 代表字節長度。

#CHAR類型
對於 CHAR 類型的字符串,MySQL 數據庫會自動對存儲列的右邊進行填充(Right Padded)操作,直到字符串達到指定的長度 N。而在讀取該列時,MySQL 數據庫會自動將 填充的字符刪除。有一種情況例外,那就是顯式地將 SQL_MODE 設置為 PAD_CHAR_TO_ FULL_LENGTH,例如:
mysql> CREATE TABLE t ( a CHAR(10));
      Query OK, 0 rows affected (0.03 sec)
mysql> INSERT INTO t SELECT abc;
      Query OK, 1 row affected (0.03 sec)
      Records: 1  Duplicates: 0  Warnings: 0
mysql> SELECT a,HEX(a),LENGTH(a) FROM t\G;
      *************************** 1. row ***************************
              a: abc
         HEX(a): 616263
      LENGTH (a): 3
      1 row in set (0.00 sec)
      mysql> SET SQL_MODE=PAD_CHAR_TO_FULL_LENGTH;
      Query OK, 0 rows affected (0.00 sec)
mysql> SELECT a,HEX(a),LENGTH(a) FROM t\G;
      *************************** 1. row ***************************
              a: abc
         HEX(a): 61626320202020202020
      LENGTH (a): 10
      1 row in set (0.00 sec)

在上述這個例子中,先創建了一張表 t,a 列的類型為 CHAR(10)。然後通過 INSERT語句插入值“abc”,因為 a 列的類型為 CHAR 型,所以會自動在後面填充空字符串,使其長 度為 10。接下來在通過 SELECT 語句取出數據時會將 a 列右填充的空字符移除,從而得到 值“abc”。通過 LENGTH 函數看到 a 列的字符長度為 3 而非 10。
接著我們將 SQL_MODE 顯式地設置為 PAD_CHAR_TO_FULL_LENGTH。這時再通過 SELECT 語句進行查詢時,得到的結果是“abc ”,abc 右邊有 7 個填充字符 0x20,並通 過 HEX 函數得到了驗證。這次 LENGTH 函數返回的長度為 10。需要註意的是,LENGTH 函數返回的是字節長度,而不是字符長度。對於多字節字符集,CHAR(N)長度的列最多 可占用的字節數為該字符集單字符最大占用字節數 *N。例如,對於 utf8 下,CHAR(10)最 多可能占用 30 個字節。通過對多字節字符串使用 CHAR_LENGTH 函數和 LENGTH 函數, 可以發現兩者的不同,示例如下:
mysql> SET NAMES gbk;
     Query OK, 0 rows affected (0.03 sec)
mysql> SELECT @a:=MySQL 技術內幕 ; Query OK, 0 rows affected (0.03 sec)
mysql> SELECT @a,HEX(@a),LENGTH(@a),CHAR_LENGTH(@a)\G; ***************************** 1. row **************************** a: MySQL 技術內幕
HEX(a): 4D7953514CBCBCCAF5C4DAC4BB
LENGTH (a): 13
CHAR_LENGTH(a): 9
1 row in set (0.00 sec)

變 量 @ a 是 g b k 字 符 集 的 字 符 串 類 型 , 值 為 “ M y S Q L 技 術 內 幕 ”, 十 六 進 制 為 0x4D7953514CBCBCCAF5C4DAC4BB,LENGTH 函數返回 13,即該字符串占用 13 字節, 因為 gbk 字符集中的中文字符占用兩個字節,因此一共占用 13 字節。CHAR_LENGTH 函數 返回 9,很顯然該字符長度為 9#VARCHAR類型
VARCHAR 類型存儲變長字段的字符類型,與 CHAR 類型不同的是,其存儲時需要在 前綴長度列表加上實際存儲的字符,該字符占用 1 ~ 2 字節的空間。當存儲的字符串長度小 於 255 字節時,其需要 1 字節的空間,當大於 255 字節時,需要 2 字節的空間。所以,對 於單字節的 latin1 來說,CHAR(10)和 VARCHAR(10)最大占用的存儲空間是不同的, CHAR(10)占用 10 個字節這是毫無疑問的,而 VARCHAR(10)的最大占用空間數是 11 字節,因為其需要 1 字節來存放字符長度。
-------------------------------------------------
註意 對於有些多字節的字符集類型,其 CHAR 和 VARCHAR 在存儲方法上是一樣的,同樣 需要為長度列表加上字符串的值。對於 GBK 和 UTF-8 這些字符類型,其有些字符是以 1 字節 存放的,有些字符是按 2 或 3 字節存放的,因此同樣需要 1 ~ 2 字節的空間來存儲字符的長 度。
-------------------------------------------------
雖然 CHAR 和 VARCHAR 的存儲方式不太相同,但是對於兩個字符串的比較,都只比 較其值,忽略 CHAR 值存在的右填充,即使將 SQL _MODE 設置為 PAD_CHAR_TO_FULL_ LENGTH 也一樣,例如:
mysql> CREATE TABLE t ( a CHAR(10), b VARCHAR(10));
    Query OK, 0 rows affected (0.01 sec)
mysql> INSERT INTO t SELECT a,a;
    Query OK, 1 row affected (0.00 sec)
    Records: 1  Duplicates: 0  Warnings: 0
mysql> SELECT a=b FROM t\G;
    *************************** 1. row ***************************
    a=b: 1
    1 row in set (0.00 sec)
    mysql> SET SQL_MODE=PAD_CHAR_TO_FULL_LENGTH;
    Query OK, 0 rows affected (0.00 sec)
mysql> SELECT a=b FROM t\G;
    *************************** 1. row ***************************
    a=b: 1
    1 row in set (0.00 sec)
官網詳解
ValueCHAR(4)Storage RequiredVARCHAR(4)Storage Required
‘‘ ‘ ‘ 4 bytes ‘‘ 1 byte
‘ab‘ ‘ab ‘ 4 bytes ‘ab‘ 3 bytes
‘abcd‘ ‘abcd‘ 4 bytes ‘abcd‘ 5 bytes
‘abcdefgh‘ ‘abcd‘ 4 bytes ‘abcd‘ 5 bytes

測試前了解兩個函數
length:查看字節數
char_length:查看字符數

1. char填充空格來滿足固定長度,但是在查詢時卻會很不要臉地刪除尾部的空格(裝作自己好像沒有浪費過空間一樣),然後修改sql_mode讓其現出原形

mysql> create table t1(x char(5),y varchar(5));
Query OK, 0 rows affected (0.26 sec)

#char存5個字符,而varchar存4個字符
mysql> insert into t1 values(你瞅啥 ,你瞅啥 );
Query OK, 1 row affected (0.05 sec)

mysql> SET sql_mode=‘‘;
Query OK, 0 rows affected, 1 warning (0.00 sec)

#在檢索時char很不要臉地將自己浪費的2個字符給刪掉了,裝的好像自己沒浪費過空間一樣,而varchar很老實,存了多少,就顯示多少
mysql> select x,char_length(x),y,char_length(y) from t1; 
+-----------+----------------+------------+----------------+
| x         | char_length(x) | y          | char_length(y) |
+-----------+----------------+------------+----------------+
| 你瞅啥    |              3 | 你瞅啥     |              4 |
+-----------+----------------+------------+----------------+
1 row in set (0.00 sec)

#略施小計,讓char現出原形
mysql> SET sql_mode = PAD_CHAR_TO_FULL_LENGTH;
Query OK, 0 rows affected (0.00 sec)

#這下子char原形畢露了......
mysql> select x,char_length(x),y,char_length(y) from t1;
+-------------+----------------+------------+----------------+
| x           | char_length(x) | y          | char_length(y) |
+-------------+----------------+------------+----------------+
| 你瞅啥      |              5 | 你瞅啥     |              4 |
+-------------+----------------+------------+----------------+
1 row in set (0.00 sec)


#char類型:3個中文字符+2個空格=11Bytes
#varchar類型:3個中文字符+1個空格=10Bytes
mysql> select x,length(x),y,length(y) from t1;
+-------------+-----------+------------+-----------+
| x           | length(x) | y          | length(y) |
+-------------+-----------+------------+-----------+
| 你瞅啥      |        11 | 你瞅啥     |        10 |
+-------------+-----------+------------+-----------+
1 row in set (0.00 sec)
技術分享
mysql> select concat(數據: ,x,長度: ,char_length(x)),concat(y,char_length(y)
) from t1;
+------------------------------------------------+--------------------------+
| concat(數據: ,x,長度: ,char_length(x))     | concat(y,char_length(y)) |
+------------------------------------------------+--------------------------+
| 數據: 你瞅啥  長度: 5                          | 你瞅啥 4                 |
+------------------------------------------------+--------------------------+
1 row in set (0.00 sec)
concat

2. 雖然 CHAR 和 VARCHAR 的存儲方式不太相同,但是對於兩個字符串的比較,都只比 較其值,忽略 CHAR 值存在的右填充,即使將 SQL _MODE 設置為 PAD_CHAR_TO_FULL_ LENGTH 也一樣,,但這不適用於like

技術分享
Values in CHAR and VARCHAR columns are sorted and compared according to the character set collation assigned to the column.

All MySQL collations are of type PAD SPACE. This means that all CHAR, VARCHAR, and TEXT values are compared without regard to any trailing spaces. “Comparison” in this context does not include the LIKE pattern-matching operator, for which trailing spaces are significant. For example:

mysql> CREATE TABLE names (myname CHAR(10));
Query OK, 0 rows affected (0.03 sec)

mysql> INSERT INTO names VALUES (Monty);
Query OK, 1 row affected (0.00 sec)

mysql> SELECT myname = Monty, myname = Monty   FROM names;
+------------------+--------------------+
| myname = Monty | myname = Monty   |
+------------------+--------------------+
|                1 |                  1 |
+------------------+--------------------+
1 row in set (0.00 sec)

mysql> SELECT myname LIKE Monty, myname LIKE Monty   FROM names;
+---------------------+-----------------------+
| myname LIKE Monty | myname LIKE Monty   |
+---------------------+-----------------------+
|                   1 |                     0 |
+---------------------+-----------------------+
1 row in set (0.00 sec)
View Code

3. 總結

#常用字符串系列:char與varchar
註:雖然varchar使用起來較為靈活,但是從整個系統的性能角度來說,char數據類型的處理速度更快,有時甚至可以超出varchar處理速度的50%。因此,用戶在設計數據庫時應當綜合考慮各方面的因素,以求達到最佳的平衡

#其他字符串系列(效率:char>varchar>text)
TEXT系列 TINYTEXT TEXT MEDIUMTEXT LONGTEXT
BLOB 系列    TINYBLOB BLOB MEDIUMBLOB LONGBLOB 
BINARY系列 BINARY VARBINARY

text:text數據類型用於保存變長的大字符串,可以組多到65535 (2**16 ? 1)個字符。
mediumtext:A TEXT column with a maximum length of 16,777,215 (2**24 ? 1) characters.
longtext:A TEXT column with a maximum length of 4,294,967,295 or 4GB (2**32 ? 1) char

五 枚舉類型與集合類型

字段的值只能在給定範圍中選擇,如單選框,多選框
enum 單選 只能在給定的範圍內選一個值,如性別 sex 男male/女female
set 多選 在給定的範圍內可以選擇一個或一個以上的值(愛好1,愛好2,愛好3...)

              枚舉類型(enum)
            An ENUM column can have a maximum of 65,535 distinct elements. (The practical limit is less than 3000.)
            示例:
                CREATE TABLE shirts (
                    name VARCHAR(40),
                    size ENUM(x-small, small, medium, large, x-large)
                );
                INSERT INTO shirts (name, size) VALUES (dress shirt,large), (t-shirt,medium),(polo shirt,small);

  

            集合類型(set)
            A SET column can have a maximum of 64 distinct members.
            示例:
                CREATE TABLE myset (col SET(a, b, c, d));
                INSERT INTO myset (col) VALUES (a,d), (d,a), (a,d,a), (a,d,d), (d,a,d);
mysql> create table student1(     #創建一個表
    -> id int primary key auto_increment,
    -> name char(5),
    -> sex enum(male,female),       #枚舉
    -> hobbies set(music,read,study,coding)   #集合
    -> );
Query OK, 0 rows affected (0.00 sec)


mysql> insert into student1(name,sex,hobbies) values(egon,None,sdsdsdsd);
Query OK, 1 row affected, 2 warnings (0.00 sec)

結果:

mysql> select * from student1;        
+----+------+------+---------+
| id | name | sex  | hobbies |
+----+------+------+---------+  
|  1 | egon |      |         |                 #可以看到不是在枚舉和集合裏的數據都會丟棄
|  2 | egon |      |         |
+----+------+------+---------+
2 rows in set (0.00 sec)


mysql> insert into student1(name,sex,hobbies) values(egon,male,music,read);
Query OK, 1 row affected (0.00 sec)

mysql> select * from student1;
+----+------+------+------------+
| id | name | sex  | hobbies    |
+----+------+------+------------+
|  1 | egon |      |            |
|  2 | egon |      |            |
|  3 | egon | male | music,read |           #有了數據了
+----+------+------+------------+
3 rows in set (0.00 sec)


總結:枚舉是單選的只能選一個,集合可以選擇多個,如果沒按照枚舉或者集合裏面的數據來插入會把數據丟棄掉

mysql數據庫從刪庫到跑路之mysql數據類型