国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 數(shù)據(jù)庫 > MySQL > 正文

MySQL數(shù)值類型溢出的處理方法

2020-01-18 20:43:24
字體:
供稿:網(wǎng)友

來,考考大家一個(gè)問題,在 MySQL 中當(dāng)某一列設(shè)置為 int(0) 時(shí)會(huì)發(fā)生什么 ?

為了演示這個(gè)問題,我們先要?jiǎng)?chuàng)建一個(gè)表

DROP TABLE IF EXISTS `na`;CREATE TABLE `na` (n1 INT(0) NOT NULL DEFAULT '0',n2 INT(11) NOT NULL DEFAULT '0');

然后我們使用下面的語句往 na 表中插入一些數(shù)據(jù)

mysql> INSERT INTO `na` VALUES(520,520),(5201314,5201314);Query OK, 2 rows affected (0.02 sec)Records: 2 Duplicates: 0 Warnings: 0

最后我們讀取出來看看

mysql> SELECT * FROM na;+---------+---------+| n1 | n2 |+---------+---------+| 520 | 520 || 5201314 | 5201314 |+---------+---------+2 rows in set (0.00 sec)

對(duì)的,好像什么都不會(huì)發(fā)生,沒什么問題才是對(duì)的,我就怕有什么問題…哈哈

我們這一章節(jié)來講講整型溢出問題。

MySQL 數(shù)值類型溢出處理

當(dāng) MySQL 在某個(gè)數(shù)值列上存儲(chǔ)超出列數(shù)據(jù)類型允許范圍的值時(shí),結(jié)果取決于當(dāng)時(shí)生效的 SQL 模式

  • 如果啟用了嚴(yán)格的 SQL 模式,則 MySQL 會(huì)根據(jù) SQL 標(biāo)準(zhǔn)拒絕帶有錯(cuò)誤的超出范圍的值,并且插入失敗
  • 如果沒有啟用任何限制模式,那么 MySQL 會(huì)將值裁剪到列數(shù)據(jù)類型范圍的上下限值并存儲(chǔ)
    • 當(dāng)超出范圍的值分配給整數(shù)列時(shí),MySQL 會(huì)存儲(chǔ)表示列數(shù)據(jù)類型范圍的相應(yīng)端點(diǎn)的值
    • 當(dāng)為浮點(diǎn)或定點(diǎn)列分配的值超出指定(或默認(rèn))精度和比例所隱含的范圍時(shí),MySQL 會(huì)存儲(chǔ)表示該范圍的相應(yīng)端點(diǎn)的值

這個(gè),應(yīng)該很好理解吧?

我們舉一個(gè)例子,假設(shè) t1 表的結(jié)構(gòu)如下

CREATE TABLE t1 (i1 TINYINT,i2 TINYINT UNSIGNED);

如果啟用了嚴(yán)格的 SQL 模式,超出范圍會(huì)發(fā)生一個(gè)錯(cuò)誤

mysql> SET sql_mode = 'TRADITIONAL'; -- 首先設(shè)置嚴(yán)格模式mysql> INSERT INTO t1 (i1, i2) VALUES(256, 256);ERROR 1264 (22003): Out of range value for column 'i1' at row 1mysql> SELECT * FROM t1;Empty set (0.00 sec)

當(dāng)嚴(yán)格模式被禁用,值可以插入,但會(huì)被裁剪,并且引發(fā)一個(gè)警告

mysql> SET sql_mode = ''; -- 禁用所有模式mysql> INSERT INTO t1 (i1, i2) VALUES(256, 256);mysql> SHOW WARNINGS;+---------+------+---------------------------------------------+| Level | Code | Message |+---------+------+---------------------------------------------+| Warning | 1264 | Out of range value for column 'i1' at row 1 || Warning | 1264 | Out of range value for column 'i2' at row 1 |+---------+------+---------------------------------------------+mysql> SELECT * FROM t1;+------+------+| i1 | i2 |+------+------+| 127 | 255 |+------+------+

如果未啟用嚴(yán)格 SQL 模式,對(duì)于 ALTER TABLE,LOAD DATA INFILE,UPDATE 和多行 INSERT等語句會(huì)由于裁剪而發(fā)生的列分配轉(zhuǎn)換并且引發(fā)一個(gè)警告。

而如果啟用了嚴(yán)格模式,這些語句會(huì)直接失敗,并且未插入或更改部分或全部值,具體取決于表是否為事務(wù)表和其他因素。

數(shù)值表達(dá)式求值過程中的溢出會(huì)導(dǎo)致錯(cuò)誤,例如,因?yàn)樽畲蟮挠蟹?hào) BIGINT 值是 9223372036854775807,因此以下表達(dá)式會(huì)產(chǎn)生錯(cuò)誤

mysql> SELECT 9223372036854775807 + 1;ERROR 1690 (22003): BIGINT value is out of range in '(9223372036854775807 + 1)'

為了在這種情況下使操作成功,需要將值轉(zhuǎn)換為 unsigned

mysql> SELECT CAST(9223372036854775807 AS UNSIGNED) + 1;+-------------------------------------------+| CAST(9223372036854775807 AS UNSIGNED) + 1 |+-------------------------------------------+| 9223372036854775808 |+-------------------------------------------+

從另一方面說,是否發(fā)生溢出取決于操作數(shù)的范圍,因此處理前一個(gè)表達(dá)式的另一種方法是使用精確值算術(shù),因?yàn)?DECIMAL 值的范圍大于整數(shù)

mysql> SELECT 9223372036854775807.0 + 1;+---------------------------+| 9223372036854775807.0 + 1 |+---------------------------+| 9223372036854775808.0 |+---------------------------+

整數(shù)數(shù)值之間的減去,如果其中一個(gè)類型為 UNSIGNED ,默認(rèn)情況下會(huì)生成無符號(hào)結(jié)果。如果為負(fù),則會(huì)引發(fā)錯(cuò)誤

mysql> SET sql_mode = '';Query OK, 0 rows affected (0.00 sec)mysql> SELECT CAST(0 AS UNSIGNED) - 1;ERROR 1690 (22003): BIGINT UNSIGNED value is out of range in '(cast(0 as unsigned) - 1)'

這種情況下,如果啟用了 NO_UNSIGNED_SUBTRACTION SQL 模式,則結(jié)果為負(fù)

mysql> SET sql_mode = 'NO_UNSIGNED_SUBTRACTION';mysql> SELECT CAST(0 AS UNSIGNED) - 1;+-------------------------+| CAST(0 AS UNSIGNED) - 1 |+-------------------------+| -1 |+-------------------------+

如果此類操作的結(jié)果用于更新 UNSIGNED 整數(shù)列,則結(jié)果將裁剪為列類型的最大值,如果啟用了 NO_UNSIGNED_SUBTRACTION 則裁剪為 0。但如果啟用了嚴(yán)格的 SQL 模式,則會(huì)發(fā)生錯(cuò)誤并且列保持不變。

后記

一切都是套路,套路….基本都和 SQL 模式有關(guān)…

總結(jié)

以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)武林網(wǎng)的支持。

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 自贡市| 且末县| 扶风县| 毕节市| 和田市| 陇南市| 陕西省| 山阳县| 蓬溪县| 林周县| 靖西县| 杭锦旗| 车致| 郓城县| 阿图什市| 台南县| 五峰| 孝昌县| 林口县| 邮箱| 瑞丽市| 咸宁市| 交城县| 安国市| 军事| 都兰县| 龙岩市| 陕西省| 江北区| 河北省| 贵港市| 濮阳县| 包头市| 永和县| 苍山县| 平舆县| 金平| 申扎县| 青州市| 山阴县| 正安县|