mysql 四 表操作
表介紹
表相當于文件,表中的一條記錄就相當于文件的一行內容,不同的是,表中的一條記錄有對應的標題,稱為表的字段
id,name,qq,age稱為字段,其余的,一行內容稱為一條記錄
本節重點:
1 創建表
2 查看表結構
3 數據類型
4 表完整性約束
5 修改表
6 復制表
7 刪除表
?
一 創建表
語法: create table 表名( 字段名1 類型[(寬度) 約束條件], 字段名2 類型[(寬度) 約束條件], 字段名3 類型[(寬度) 約束條件] );注意: 1. 在同一張表中,字段名是不能相同 2. 寬度和約束條件可選 3. 字段名和類型是必須的?
?
MariaDB [(none)]> create database db1 charset utf8;MariaDB [(none)]> use db1;MariaDB [db1]> create table t1( -> id int, -> name varchar(50),-> sex enum('male','female'),-> age int(3)-> );MariaDB [db1]> show tables; #查看db1庫下所有表名 MariaDB [db1]> desc t1; +-------+-----------------------+------+-----+---------+-------+ | Field | Type | Null | Key | Default | Extra | +-------+-----------------------+------+-----+---------+-------+ | id | int(11) | YES | | NULL | | | name | varchar(50) | YES | | NULL | | | sex | enum('male','female') | YES | | NULL | | | age | int(3) | YES | | NULL | | +-------+-----------------------+------+-----+---------+-------+MariaDB [db1]> select id,name,sex,age from t1; Empty set (0.00 sec)MariaDB [db1]> select * from t1; Empty set (0.00 sec)MariaDB [db1]> select id,name from t1; Empty set (0.00 sec)往表中插入數據
MariaDB [db1]> insert into t1 values-> (1,'egon',18,'male'),-> (2,'alex',81,'female')-> ; MariaDB [db1]> select * from t1; +------+------+------+--------+ | id | name | age | sex | +------+------+------+--------+ | 1 | egon | 18 | male | | 2 | alex | 81 | female | +------+------+------+--------+MariaDB [db1]> insert into t1(id) values -> (3),-> (4); MariaDB [db1]> select * from t1; +------+------+------+--------+ | id | name | age | sex | +------+------+------+--------+ | 1 | egon | 18 | male | | 2 | alex | 81 | female | | 3 | NULL | NULL | NULL | | 4 | NULL | NULL | NULL | +------+------+------+--------+ #亂碼問題:重要!!!mysql> create database db1 charset latin1; mysql> use db1; mysql> create table t1(name varchar(20)); mysql> show create table t1; #查看表,發現表默認與數據db1的字符編碼一致 mysql> insert into t1 values('林'); #插入中文出錯,因為latin1不支持中文 ERROR 1366 (HY000): mysql> #解決方法一:刪除庫db1,重建db1,字符編碼指定為utf8#解決方法二:修改 mysql> alter table t1 charset utf8; #修改表t1的編碼 mysql> insert into t1 values('林'); #雖然t1的編碼改了,但是t1的字段name仍然是按照latin1編碼創建的 ERROR 1366 (HY000): mysql> alter table t1 modify name varchar(20); #需要重新定義下字段name mysql> insert into t1 values('林'); mysql> select * from t1; +------+ | name | +------+ | 林 | +------+ps:不要忘記將數據庫編碼也改成utf8,這樣以后在該數據庫下創建表時,都默認utf8編碼了#配置文件:http://blog.csdn.net/yipiankongbai/article/details/16937815
永久解決編碼問題
#1. 修改配置文件 [mysqld] default-character-set=utf8 [client] default-character-set=utf8 [mysql] default-character-set=utf8#mysql5.5以上:修改方式有所改動 [mysqld]character-set-server=utf8collation-server=utf8_general_ci[client]default-character-set=utf8[mysql]default-character-set=utf8#2. 重啟服務 #3. 查看修改結果: \s show variables like '%char%'?
注意注意注意:表中的最后一個字段不要加逗號?
二 查看表結構
MariaDB [db1]> describe t1; #查看表結構,可簡寫為desc 表名 +-------+-----------------------+------+-----+---------+-------+ | Field | Type | Null | Key | Default | Extra | +-------+-----------------------+------+-----+---------+-------+ | id | int(11) | YES | | NULL | | | name | varchar(50) | YES | | NULL | | | sex | enum('male','female') | YES | | NULL | | | age | int(3) | YES | | NULL | | +-------+-----------------------+------+-----+---------+-------+MariaDB [db1]> show create table t1\G; #查看表詳細結構,可加\G三 數據類型
一 介紹
存儲引擎決定了表的類型,而表內存放的數據也要有不同的類型,每種數據類型都有自己的寬度,但寬度是可選的
詳細參考:
- http://www.runoob.com/mysql/mysql-data-types.html
- http://dev.mysql.com/doc/refman/5.7/en/data-type-overview.html
mysql數據類型概覽
#1. 數字: 整型:tinyinit int bigint小數:float :在位數比較短的情況下不精準double :在位數比較長的情況下不精準0.000001230123123123存成:0.000001230000decimal:(如果用小數,則用推薦使用decimal)精準內部原理是以字符串形式去存#2. 字符串:char(10):簡單粗暴,浪費空間,存取速度快root存成root000000varchar:精準,節省空間,存取速度慢sql優化:創建表時,定長的類型往前放,變長的往后放比如性別 比如地址或描述信息>255個字符,超了就把文件路徑存放到數據庫中。比如圖片,視頻等找一個文件服務器,數據庫中只存路徑或url。#3. 時間類型: 最常用:datetime#4. 枚舉類型與集合類型二 數值類型
? ? ? 整數類型:TINYINT SMALLINT MEDIUMINT INT BIGINT
作用:存儲年齡,等級,id,各種號碼等
========================================tinyint[(m)] [unsigned] [zerofill]小整數,數據類型用于保存一些范圍的整數數值范圍:有符號:-128 ~ 127無符號: ~ 255PS: MySQL中無布爾值,使用tinyint(1)構造。========================================int[(m)][unsigned][zerofill]整數,數據類型用于保存一些范圍的整數數值范圍:有符號:-2147483648 ~ 2147483647無符號: ~ 4294967295========================================bigint[(m)][unsigned][zerofill]大整數,數據類型用于保存一些范圍的整數數值范圍:有符號:-9223372036854775808 ~ 9223372036854775807無符號:~ 18446744073709551615 =========有符號和無符號tinyint========== #tinyint默認為有符號 MariaDB [db1]> create table t1(x tinyint); #默認為有符號,即數字前有正負號 MariaDB [db1]> desc t1; MariaDB [db1]> insert into t1 values-> (-129),-> (-128),-> (127),-> (128); MariaDB [db1]> select * from t1; +------+ | x | +------+ | -128 | #-129存成了-128 | -128 | #有符號,最小值為-128 | 127 | #有符號,最大值127 | 127 | #128存成了127 +------+#設置無符號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,1.175494351E-38 to 3.402823466E+38無符號:1.175494351E-38 to 3.402823466E+38**** 數值越大,越不準確 ****======================================DOUBLE[(M,D)] [UNSIGNED] [ZEROFILL]雙精度浮點數(非準確小數值),m是數字總個數,d是小數點后個數。有符號:-1.7976931348623157E+308 to -2.2250738585072014E-3082.2250738585072014E-308 to 1.7976931348623157E+308無符號:2.2250738585072014E-308 to 1.7976931348623157E+308**** 數值越大,越不準確 ****驗證
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()顯示為十六進制
驗證
三 日期類型
DATE TIME DATETIME?TIMESTAMP YEAR?
作用:存儲用戶注冊時間,文章發布時間,員工入職時間,出生時間,過期時間等
YEARYYYY(1901/2155)DATEYYYY-MM-DD(1000-01-01/9999-12-31)TIMEHH:MM:SS('-838:59:59'/'838:59:59')DATETIMEYYYY-MM-DD HH:MM:SS(1000-01-01 00:00:00/9999-12-31 23:59:59 Y)TIMESTAMPYYYYMMDD 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 | +------+------+-----------+------------+------------+---------------------+
datetime與timestamp的區別
在實際應用的很多場景中,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語句中沒有指定該列的更新值,則默認更新為當前時間。?
四 字符串類型
#官網: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: abcHEX(a): 616263LENGTH (a): 3 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: abcHEX(a): 61626320202020202020LENGTH (a): 10 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 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 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 row in set (0.00 sec)| '' | '????' | 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 | +-----------+----------------+------------+----------------+ 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 | +-------------+----------------+------------+----------------+ 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 | +-------------+-----------+------------+-----------+ row in set (0.00 sec) #了解concat 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 | +------------------------------------------------+--------------------------+ row in set (0.00 sec)?
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 | +------------------+--------------------+ row in set (0.00 sec)mysql> SELECT myname LIKE 'Monty', myname LIKE 'Monty ' FROM names; +---------------------+-----------------------+ | myname LIKE 'Monty' | myname LIKE 'Monty ' | +---------------------+-----------------------+ | 1 | 0 | +---------------------+-----------------------+ row in set (0.00 sec)3. 總結
#常用字符串系列:char與varchar 注:雖然varchar使用起來較為靈活,但是從整個系統的性能角度來說,char數據類型的處理速度更快,有時甚至可以超出varchar處理速度的50%。因此,用戶在設計數據庫時應當綜合考慮各方面的因素,以求達到最佳的平衡#其他字符串系列(效率:char>varchar>text) TEXT系列 TINYTEXT TEXT MEDIUMTEXT LONGTEXT BLOB 系列 TINYBLOB BLOB MEDIUMBLOB LONGBLOB BINARY系列 BINARY VARBINARYtext: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) characters.五 枚舉類型與集合類型
字段的值只能在給定范圍中選擇,如單選框,多選框
enum 單選 只能在給定的范圍內選一個值,如性別 sex 男male/女female
set 多選 在給定的范圍內可以選擇一個或一個以上的值(愛好1,愛好2,愛好3...)
?
四 表完整性約束
一 介紹
約束條件與數據類型的寬度一樣,都是可選參數
作用:用于保證數據的完整性和一致性
主要分為:
?
PRIMARY KEY (PK) 標識該字段為該表的主鍵,可以唯一的標識記錄 FOREIGN KEY (FK) 標識該字段為該表的外鍵 NOT NULL 標識該字段不能為空 UNIQUE KEY (UK) 標識該字段的值是唯一的 AUTO_INCREMENT 標識該字段的值自動增長(整數類型,而且為主鍵) DEFAULT 為該字段設置默認值UNSIGNED 無符號 ZEROFILL 使用0填充說明:
1. 是否允許為空,默認NULL,可設置NOT NULL,字段不允許為空,必須賦值 2. 字段是否有默認值,缺省的默認值是NULL,如果插入記錄時不給字段賦值,此字段使用默認值 sex enum('male','female') not null default 'male' age int unsigned NOT NULL default 20 必須為正值(無符號) 不允許為空 默認是20 3. 是否是key 主鍵 primary key 外鍵 foreign key 索引 (index,unique...)二 not null與default
是否可空,null表示空,非字符串
not null - 不可空
null - 可空
默認值,創建列時可以指定默認值,當插入數據時如果未主動設置,則自動添加默認值
create table tb1(
nid int not null defalut 2,
num int not null
)
驗證
三 unique
============設置唯一約束 UNIQUE=============== 方法一: create table department1( id int, name varchar(20) unique, comment varchar(100) );方法二: create table department2( id int, name varchar(20), comment varchar(100), constraint uk_name unique(name) );mysql> insert into department1 values(1,'IT','技術'); Query OK, 1 row affected (0.00 sec) mysql> insert into department1 values(1,'IT','技術'); ERROR 1062 (23000): Duplicate entry 'IT' for key 'name'not null+unique的化學反應
mysql> create table t1(id int not null unique); Query OK, 0 rows affected (0.02 sec)mysql> desc t1; +-------+---------+------+-----+---------+-------+ | Field | Type | Null | Key | Default | Extra | +-------+---------+------+-----+---------+-------+ | id | int(11) | NO | PRI | NULL | | +-------+---------+------+-----+---------+-------+ row in set (0.00 sec) #聯合唯一create table service( id int primary key auto_increment, name varchar(20), host varchar(15) not null, port int not null, unique(host,port) #聯合唯一 );mysql> insert into service values-> (1,'nginx','192.168.0.10',80),-> (2,'haproxy','192.168.0.20',80),-> (3,'mysql','192.168.0.30',3306)-> ; Query OK, 3 rows affected (0.01 sec) Records: 3 Duplicates: 0 Warnings: 0mysql> insert into service(name,host,port) values('nginx','192.168.0.10',80); ERROR 1062 (23000): Duplicate entry '192.168.0.10-80' for key 'host'
四 primary key
primary key字段的值不為空且唯一
一個表中可以:
單列做主鍵
多列做主鍵(復合主鍵)
但一個表內只能有一個主鍵primary key
?
單列主鍵
============單列做主鍵=============== #方法一:not null+unique create table department1( id int not null unique, #主鍵 name varchar(20) not null unique, comment varchar(100) );mysql> desc department1; +---------+--------------+------+-----+---------+-------+ | Field | Type | Null | Key | Default | Extra | +---------+--------------+------+-----+---------+-------+ | id | int(11) | NO | PRI | NULL | | | name | varchar(20) | NO | UNI | NULL | | | comment | varchar(100) | YES | | NULL | | +---------+--------------+------+-----+---------+-------+ rows in set (0.01 sec)#方法二:在某一個字段后用primary key create table department2( id int primary key, #主鍵 name varchar(20), comment varchar(100) );mysql> desc department2; +---------+--------------+------+-----+---------+-------+ | Field | Type | Null | Key | Default | Extra | +---------+--------------+------+-----+---------+-------+ | id | int(11) | NO | PRI | NULL | | | name | varchar(20) | YES | | NULL | | | comment | varchar(100) | YES | | NULL | | +---------+--------------+------+-----+---------+-------+ rows in set (0.00 sec)#方法三:在所有字段后單獨定義primary key create table department3( id int, name varchar(20), comment varchar(100), constraint pk_name primary key(id); #創建主鍵并為其命名pk_name mysql> desc department3; +---------+--------------+------+-----+---------+-------+ | Field | Type | Null | Key | Default | Extra | +---------+--------------+------+-----+---------+-------+ | id | int(11) | NO | PRI | NULL | | | name | varchar(20) | YES | | NULL | | | comment | varchar(100) | YES | | NULL | | +---------+--------------+------+-----+---------+-------+ rows in set (0.01 sec)?
多列主鍵
==================多列做主鍵================ create table service( ip varchar(15), port char(5), service_name varchar(10) not null, primary key(ip,port) );mysql> desc service; +--------------+-------------+------+-----+---------+-------+ | Field | Type | Null | Key | Default | Extra | +--------------+-------------+------+-----+---------+-------+ | ip | varchar(15) | NO | PRI | NULL | | | port | char(5) | NO | PRI | NULL | | | service_name | varchar(10) | NO | | NULL | | +--------------+-------------+------+-----+---------+-------+ rows in set (0.00 sec)mysql> insert into service values-> ('172.16.45.10','3306','mysqld'),-> ('172.16.45.11','3306','mariadb')-> ; Query OK, 2 rows affected (0.00 sec) Records: 2 Duplicates: 0 Warnings: 0mysql> insert into service values ('172.16.45.10','3306','nginx'); ERROR 1062 (23000): Duplicate entry '172.16.45.10-3306' for key 'PRIMARY'?
?
五 auto_increment
約束字段為自動增長,被約束的字段必須同時被key約束
#不指定id,則自動增長 create table student( id int primary key auto_increment, name varchar(20), sex enum('male','female') default 'male' );mysql> desc student; +-------+-----------------------+------+-----+---------+----------------+ | Field | Type | Null | Key | Default | Extra | +-------+-----------------------+------+-----+---------+----------------+ | id | int(11) | NO | PRI | NULL | auto_increment | | name | varchar(20) | YES | | NULL | | | sex | enum('male','female') | YES | | male | | +-------+-----------------------+------+-----+---------+----------------+ mysql> insert into student(name) values-> ('egon'),-> ('alex')-> ;mysql> select * from student; +----+------+------+ | id | name | sex | +----+------+------+ | 1 | egon | male | | 2 | alex | male | +----+------+------+#也可以指定id mysql> insert into student values(4,'asb','female'); Query OK, 1 row affected (0.00 sec)mysql> insert into student values(7,'wsb','female'); Query OK, 1 row affected (0.00 sec)mysql> select * from student; +----+------+--------+ | id | name | sex | +----+------+--------+ | 1 | egon | male | | 2 | alex | male | | 4 | asb | female | | 7 | wsb | female | +----+------+--------+#對于自增的字段,在用delete刪除后,再插入值,該字段仍按照刪除前的位置繼續增長 mysql> delete from student; Query OK, 4 rows affected (0.00 sec)mysql> select * from student; Empty set (0.00 sec)mysql> insert into student(name) values('ysb'); mysql> select * from student; +----+------+------+ | id | name | sex | +----+------+------+ | 8 | ysb | male | +----+------+------+#應該用truncate清空表,比起delete一條一條地刪除記錄,truncate是直接清空表,在刪除大表時用它 mysql> truncate student; Query OK, 0 rows affected (0.01 sec)mysql> insert into student(name) values('egon'); Query OK, 1 row affected (0.01 sec)mysql> select * from student; +----+------+------+ | id | name | sex | +----+------+------+ | 1 | egon | male | +----+------+------+ row in set (0.00 sec) View Code #在創建完表后,修改自增字段的起始值 mysql> create table student(-> id int primary key auto_increment,-> name varchar(20),-> sex enum('male','female') default 'male'-> );mysql> alter table student auto_increment=3;mysql> show create table student; ....... ENGINE=InnoDB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8mysql> insert into student(name) values('egon'); Query OK, 1 row affected (0.01 sec)mysql> select * from student; +----+------+------+ | id | name | sex | +----+------+------+ | 3 | egon | male | +----+------+------+ row in set (0.00 sec)mysql> show create table student; ....... ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8#也可以創建表時指定auto_increment的初始值,注意初始值的設置為表選項,應該放到括號外 create table student( id int primary key auto_increment, name varchar(20), sex enum('male','female') default 'male' )auto_increment=3;#設置步長 sqlserver:自增步長基于表級別create table t1(id int。。。)engine=innodb,auto_increment=2 步長=2 default charset=utf8mysql自增的步長:show session variables like 'auto_inc%';#基于會話級別set session auth_increment_increment=2 #修改會話級別的步長#基于全局級別的set global auth_increment_increment=2 #修改全局級別的步長(所有會話都生效)#!!!注意了注意了注意了!!! If the value of auto_increment_offset is greater than that of auto_increment_increment, the value of auto_increment_offset is ignored. 翻譯:如果auto_increment_offset的值大于auto_increment_increment的值,則auto_increment_offset的值會被忽略 比如:設置auto_increment_offset=3,auto_increment_increment=2mysql> set global auto_increment_increment=5; Query OK, 0 rows affected (0.00 sec)mysql> set global auto_increment_offset=3; Query OK, 0 rows affected (0.00 sec)mysql> show variables like 'auto_incre%'; #需要退出重新登錄 +--------------------------+-------+ | Variable_name | Value | +--------------------------+-------+ | auto_increment_increment | 1 | | auto_increment_offset | 1 | +--------------------------+-------+create table student( id int primary key auto_increment, name varchar(20), sex enum('male','female') default 'male' );mysql> insert into student(name) values('egon1'),('egon2'),('egon3'); mysql> select * from student; +----+-------+------+ | id | name | sex | +----+-------+------+ | 3 | egon1 | male | | 8 | egon2 | male | | 13 | egon3 | male | +----+-------+------+ 步長increment與起始偏移量offset:auto_increment_increment,auto_increment_offset六 foreign key
員工信息表有三個字段:工號 ?姓名 ?部門
公司有3個部門,但是有1個億的員工,那意味著部門這個字段需要重復存儲,部門名字越長,越浪費
解決方法:
我們完全可以定義一個部門表
然后讓員工信息表關聯該表,如何關聯,即foreign key
#表類型必須是innodb存儲引擎,且被關聯的字段,即references指定的另外一個表的字段,必須保證唯一 create table department( id int primary key, name varchar(20) not null )engine=innodb;#dpt_id外鍵,關聯父表(department主鍵id),同步更新,同步刪除 create table employee( id int primary key, name varchar(20) not null, dpt_id int, constraint fk_name foreign key(dpt_id) references department(id) on delete cascade on update cascade )engine=innodb;#先往父表department中插入記錄 insert into department values (1,'歐德博愛技術有限事業部'), (2,'艾利克斯人力資源部'), (3,'銷售部');#再往子表employee中插入記錄 insert into employee values (1,'egon',1), (2,'alex1',2), (3,'alex2',2), (4,'alex3',2), (5,'李坦克',3), (6,'劉飛機',3), (7,'張火箭',3), (8,'林子彈',3), (9,'加特林',3) ;#刪父表department,子表employee中對應的記錄跟著刪 mysql> delete from department where id=3; mysql> select * from employee; +----+-------+--------+ | id | name | dpt_id | +----+-------+--------+ | 1 | egon | 1 | | 2 | alex1 | 2 | | 3 | alex2 | 2 | | 4 | alex3 | 2 | +----+-------+--------+#更新父表department,子表employee中對應的記錄跟著改 mysql> update department set id=22222 where id=2; mysql> select * from employee; +----+-------+--------+ | id | name | dpt_id | +----+-------+--------+ | 1 | egon | 1 | | 3 | alex2 | 22222 | | 4 | alex3 | 22222 | | 5 | alex1 | 22222 | +----+-------+--------+ 表1 foreign key 表2 則表1的多條記錄對應表2的一條記錄,即多對一利用foreign key的原理我們可以制作兩張表的多對多,一對一關系 多對多:表1的多條記錄可以對應表2的一條記錄表2的多條記錄也可以對應表1的一條記錄一對一:表1的一條記錄唯一對應表2的一條記錄,反之亦然分析時,我們先從按照上面的基本原理去套,然后再翻譯成真實的意義,就很好理解了 輔助理解 #一對多或稱為多對一 三張表:出版社,作者信息,書一對多(或多對一):一個出版社可以出版多本書關聯方式:foreign key =====================多對一===================== create table press( id int primary key auto_increment, name varchar(20) );create table book( id int primary key auto_increment, name varchar(20), press_id int not null, foreign key(press_id) references press(id) on delete cascade on update cascade );insert into press(name) values ('北京工業地雷出版社'), ('人民音樂不好聽出版社'), ('知識產權沒有用出版社') ;insert into book(name,press_id) values ('九陽神功',1), ('九陰真經',2), ('九陰白骨爪',2), ('獨孤九劍',3), ('降龍十巴掌',2), ('葵花寶典',3) ; View Code
#多對多 三張表:出版社,作者信息,書 多對多:一個作者可以寫多本書,一本書也可以有多個作者,雙向的一對多,即多對多 關聯方式:foreign key+一張新的表
?
=====================多對多===================== create table author( id int primary key auto_increment, name varchar(20) );#這張表就存放作者表與書表的關系,即查詢二者的關系查這表就可以了 create table author2book( id int not null unique auto_increment, author_id int not null, book_id int not null, constraint fk_author foreign key(author_id) references author(id) on delete cascade on update cascade, constraint fk_book foreign key(book_id) references book(id) on delete cascade on update cascade, primary key(author_id,book_id) );#插入四個作者,id依次排開 insert into author(name) values('egon'),('alex'),('yuanhao'),('wpq');#每個作者與自己的代表作如下 egon: 九陽神功 九陰真經 九陰白骨爪 獨孤九劍 降龍十巴掌 葵花寶典 alex: 九陽神功 葵花寶典 yuanhao: 獨孤九劍 降龍十巴掌 葵花寶典 wpq: 九陽神功insert into author2book(author_id,book_id) values (1,1), (1,2), (1,3), (1,4), (1,5), (1,6), (2,1), (2,6), (3,4), (3,5), (3,6), (4,1) ; View Code?
?
#一對一 兩張表:學生表和客戶表一對一:一個學生是一個客戶,一個客戶有可能變成一個學校,即一對一的關系關聯方式:foreign key+unique ?View Code七 作業
練習:賬號信息表,用戶組,主機表,主機組
#用戶表 create table user( id int not null unique auto_increment, username varchar(20) not null, password varchar(50) not null, primary key(username,password) );insert into user(username,password) values ('root','123'), ('egon','456'), ('alex','alex3714') ;#用戶組表 create table usergroup( id int primary key auto_increment, groupname varchar(20) not null unique );insert into usergroup(groupname) values ('IT'), ('Sale'), ('Finance'), ('boss') ;#主機表 create table host( id int primary key auto_increment, ip char(15) not null unique default '127.0.0.1' );insert into host(ip) values ('172.16.45.2'), ('172.16.31.10'), ('172.16.45.3'), ('172.16.31.11'), ('172.10.45.3'), ('172.10.45.4'), ('172.10.45.5'), ('192.168.1.20'), ('192.168.1.21'), ('192.168.1.22'), ('192.168.2.23'), ('192.168.2.223'), ('192.168.2.24'), ('192.168.3.22'), ('192.168.3.23'), ('192.168.3.24') ;#業務線表 create table business( id int primary key auto_increment, business varchar(20) not null unique ); insert into business(business) values ('輕松貸'), ('隨便花'), ('大富翁'), ('窮一生') ;#建關系:user與usergroup create table user2usergroup( id int not null unique auto_increment, user_id int not null, group_id int not null, primary key(user_id,group_id), foreign key(user_id) references user(id), foreign key(group_id) references usergroup(id) );insert into user2usergroup(user_id,group_id) values (1,1), (1,2), (1,3), (1,4), (2,3), (2,4), (3,4) ;#建關系:host與business create table host2business( id int not null unique auto_increment, host_id int not null, business_id int not null, primary key(host_id,business_id), foreign key(host_id) references host(id), foreign key(business_id) references business(id) );insert into host2business(host_id,business_id) values (1,1), (1,2), (1,3), (2,2), (2,3), (3,4) ;#建關系:user與host create table user2host( id int not null unique auto_increment, user_id int not null, host_id int not null, primary key(user_id,host_id), foreign key(user_id) references user(id), foreign key(host_id) references host(id) );insert into user2host(user_id,host_id) values (1,1), (1,2), (1,3), (1,4), (1,5), (1,6), (1,7), (1,8), (1,9), (1,10), (1,11), (1,12), (1,13), (1,14), (1,15), (1,16), (2,2), (2,3), (2,4), (2,5), (3,10), (3,11), (3,12) ;作業:
?
五 修改表ALTER TABLE
語法: 1. 修改表名ALTER TABLE 表名 RENAME 新表名;2. 增加字段ALTER TABLE 表名ADD 字段名 數據類型 [完整性約束條件…],ADD 字段名 數據類型 [完整性約束條件…];ALTER TABLE 表名ADD 字段名 數據類型 [完整性約束條件…] FIRST;ALTER TABLE 表名ADD 字段名 數據類型 [完整性約束條件…] AFTER 字段名;3. 刪除字段ALTER TABLE 表名 DROP 字段名;4. 修改字段ALTER TABLE 表名 MODIFY 字段名 數據類型 [完整性約束條件…];ALTER TABLE 表名 CHANGE 舊字段名 新字段名 舊數據類型 [完整性約束條件…];ALTER TABLE 表名 CHANGE 舊字段名 新字段名 新數據類型 [完整性約束條件…];示例: 1. 修改存儲引擎 mysql> alter table service -> engine=innodb;2. 添加字段 mysql> alter table student10-> add name varchar(20) not null,-> add age int(3) not null default 22;mysql> alter table student10-> add stu_num varchar(10) not null after name; //添加name字段之后mysql> alter table student10 -> add sex enum('male','female') default 'male' first; //添加到最前面3. 刪除字段 mysql> alter table student10-> drop sex;mysql> alter table service-> drop mac;4. 修改字段類型modify mysql> alter table student10-> modify age int(3); mysql> alter table student10-> modify id int(11) not null primary key auto_increment; //修改為主鍵5. 增加約束(針對已有的主鍵增加auto_increment) mysql> alter table student10 modify id int(11) not null primary key auto_increment; ERROR 1068 (42000): Multiple primary key definedmysql> alter table student10 modify id int(11) not null auto_increment; Query OK, 0 rows affected (0.01 sec) Records: 0 Duplicates: 0 Warnings: 06. 對已經存在的表增加復合主鍵 mysql> alter table service2-> add primary key(host_ip,port); 7. 增加主鍵 mysql> alter table student1-> modify name varchar(10) not null primary key;8. 增加主鍵和自動增長 mysql> alter table student1-> modify id int not null primary key auto_increment;9. 刪除主鍵 a. 刪除自增約束 mysql> alter table student10 modify id int(11) not null; b. 刪除主鍵 mysql> alter table student10 -> drop primary key;六 復制表
復制表結構+記錄 (key不會復制: 主鍵、外鍵和索引) mysql> create table new_service select * from service;只復制表結構 mysql> select * from service where 1=2; //條件為假,查不到任何記錄 Empty set (0.00 sec) mysql> create table new1_service select * from service where 1=2; Query OK, 0 rows affected (0.00 sec) Records: 0 Duplicates: 0 Warnings: 0mysql> create table t4 like employees;七 刪除表
DROP TABLE 表名;?
轉載于:https://www.cnblogs.com/ctztake/p/7486368.html
總結
以上是生活随笔為你收集整理的mysql 四 表操作的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 现在本田车换刹车片都换什么牌子的?
- 下一篇: 办摩托车的临时牌是否影响汽车驾驶证?