MySQL 字符串指南

字符串是你在 MySQL 中使用的最常見的數(shù)據(jù)類型之一。許多用戶在他們的數(shù)據(jù)庫(kù)中插入和讀取字符串,而沒有認(rèn)真地了解過它們。本文旨在讓你深入了解 MySQL 如何存儲(chǔ)和顯示你的字符串變量,以便你能更好地控制你的數(shù)據(jù)。
你可以把字符串分成兩類:二進(jìn)制和非二進(jìn)制。你可能在大多數(shù)時(shí)候想到的是非二進(jìn)制字符串。非二進(jìn)制字符串有字符集和排序的不同。另一方面,二進(jìn)制字符串存儲(chǔ)諸如 MP3 文件或圖像等東西。即使你在二進(jìn)制字符串中存儲(chǔ)了一個(gè)詞,比如“歌曲”,它的存儲(chǔ)方式也與非二進(jìn)制字符串不同。
我將重點(diǎn)討論非二進(jìn)制字符串。MySQL 中的所有非二進(jìn)制字符串都與字符集和排序相關(guān)。字符串的字符集控制哪些字符可以存儲(chǔ)在字符串中,而它的排序方式控制當(dāng)你顯示字符串時(shí)如何排序。
字符集
要查看你系統(tǒng)中的字符集,請(qǐng)運(yùn)行以下命令:
這個(gè)命令將輸出四列數(shù)據(jù),包括字符集:
- 名稱
- 簡(jiǎn)要描述
- 默認(rèn)的排序方式
- 字符集中每個(gè)字符的最大尺寸
MySQL 過去默認(rèn)為 ??latin1?? 字符集,但自 8.0 版以來(lái),默認(rèn)為 ??utf8mb4??。現(xiàn)在的默認(rèn)排序方式是 ??utf8mb4_0900_ai_ci??。??ai?? 表示該排序?qū)σ粽{(diào)不敏感( ??á?? = ??a??),而 ??ci?? 則指定它對(duì)大小寫不敏感(??a?? = ??A??)。
不同的字符集將其字符存儲(chǔ)在內(nèi)存中不同大小的塊中。例如,從上面的命令可以看出,存儲(chǔ)在 ??utf8mb4?? 的字符被存儲(chǔ)在 1 到 4 個(gè)字節(jié)大小的內(nèi)存中。如果你想看看一個(gè)字符串是否包含多字節(jié)的字符,你可以使用 ??CHAR_LENGTH()?? 和 ??LENGTH()?? 函數(shù)。??CHAR_LENGTH()?? 顯示一個(gè)字符串包含多少個(gè)字符,而 ??LENGTH()?? 顯示一個(gè)字符串有多少個(gè)字節(jié),根據(jù)字符集的不同,它可能與一個(gè)字符串的字符長(zhǎng)度相同,也可能不相同。下面是一個(gè)例子:
這個(gè)例子表明,??latin1?? 字符集以單字節(jié)為單位存儲(chǔ)字符。其他字符集,如 ??utf16??,允許多字節(jié)的字符:
排序
當(dāng)你運(yùn)行帶有 ??ORDER BY?? 子句的 SQL 語(yǔ)句時(shí),字符串排序方式將決定值的顯示方式。你對(duì)排序方式的選擇是由你選擇的字符集決定的。當(dāng)你運(yùn)行上面的 ??SHOW CHARACTER SET?? 命令時(shí),你看到了每個(gè)字符集的默認(rèn)排序方式。你可以很容易地看到某個(gè)特定字符集的所有排序方式。例如,如果你想查看 ??utf8mb4?? 字符集允許哪些排序,請(qǐng)運(yùn)行:
排序方式可以是不區(qū)分大小寫的,也可以是區(qū)分大小寫的,或者是二進(jìn)制的。讓我們建立一個(gè)簡(jiǎn)單的表,向其中插入一些值,然后用不同的排序方式查看數(shù)據(jù),看看輸出結(jié)果有什么不同:
在不區(qū)分大小寫的情況下,你的數(shù)據(jù)會(huì)按字母順序返回,但不能保證大寫的單詞會(huì)排在小寫的單詞之前,如下圖所示:
另一方面,當(dāng) MySQL 運(yùn)行大小寫敏感的搜索時(shí),每個(gè)字母的小寫將排在大寫之前:
而按二進(jìn)制排序方式將返回所有大寫的值,然后再返回小寫的值:
如果你想知道一個(gè)字符串使用哪種字符集和排序,你可以使用被恰當(dāng)命名的 ??charset?? 和 ??collation?? 函數(shù)。運(yùn)行 MySQL 8.0 或更高版本的服務(wù)器將默認(rèn)使用 ??utf8mb4?? 字符集和 ??utf8mb4_0900_ai_ci?? 排序:
你可以使用 ??SET NAMES?? 命令來(lái)改變所使用的字符集或排序方式。
要從 ??utf8mb4?? 字符集改為 ??utf16??,運(yùn)行這個(gè)命令:
如果你想選擇默認(rèn)以外的排序方式,你可以在 ??SET NAMES?? 命令中添加一個(gè) ??COLLATE?? 子句。
例如,假設(shè)你的數(shù)據(jù)庫(kù)存儲(chǔ)西班牙語(yǔ)的單詞。MySQL 的默認(rèn)排序(??utf8mb4_0900_ai_ci??)將 ??ch?? 和 ??ll?? 視為兩個(gè)不同的字符,并將它們排序。但在西班牙語(yǔ)中,??ch?? 和 ??ll?? 是單獨(dú)的字母,所以如果你想讓它們按正確的順序排序(分別排在 ??c?? 和 ??l?? 之后),你需要使用不同的排序。一個(gè)選擇是使用 ??utf8mb4_spanish2_ci?? 排序方式:
儲(chǔ)存字符串
MySQL 允許你為你的字符串值選擇不同的數(shù)據(jù)類型。(甚至比其他流行的數(shù)據(jù)庫(kù),如 PostgreSQL 和 MongoDB 更多。)
下面是 MySQL 的二進(jìn)制字符串?dāng)?shù)據(jù)類型的列表、它們的非二進(jìn)制對(duì)應(yīng)物,以及它們的最大長(zhǎng)度:
- ?
?binary??:??char??(255) - ?
?varbinary??:??varchar??(65,535) - ?
?tinyblob??:??tinytext??(255) - ?
?blob??:??text??(65,535) - ?
?mediumblob??:??mediumtext??(16,777,215) - ?
?longblob??:??longtext??(4,294,967,295)
要記住的一件重要事情是,與被存儲(chǔ)在可變長(zhǎng)度的字段中的 ??varbinary??、??varchar??、??text?? 和 ??blob?? 類型不同(也就是說,只使用需要的空間),MySQL 將二進(jìn)制(??binary??)和字符(??char??)類型存儲(chǔ)在固定長(zhǎng)度的字段。因此,像 ??char(20)?? 或 ??binary(20)?? 這樣的值將總是占用 20 個(gè)字節(jié),即使你在其中存儲(chǔ)了少于 20 個(gè)字符。對(duì)于二進(jìn)制類型,MySQL用 ASCII NUL 值(??0x00??)填充這些值,對(duì)于 字符類型,用空格填充。
在選擇數(shù)據(jù)類型時(shí)要考慮的另一件事是,你是否希望在字符串后面的空格被保留或剝離。在顯示數(shù)據(jù)時(shí),MySQL 會(huì)從以字符數(shù)據(jù)類型存儲(chǔ)的數(shù)據(jù)中剝離空格,但不會(huì)剝離 ??varchar?? 的空格。
總結(jié)
字符串是數(shù)據(jù)庫(kù)中最常用的數(shù)據(jù)類型之一,而 MySQL 仍然是當(dāng)今最流行的數(shù)據(jù)庫(kù)系統(tǒng)之一。我希望你能從這篇文章中學(xué)到一些新的東西,并能用你的新知識(shí)來(lái)提高你的數(shù)據(jù)庫(kù)技能。



















