這是MySQL 5實力養成暨評量裡的4-30.『MySQL對於浮點數資料型態,有支援近似值(M,D)的語法,新增一筆資料時,當整數位(小數點左邊的位數)超過幾位時,MySQL會發生「#1264 – Out of range value adjusted for column」的錯誤?
答案:(C) M-D
這是第十一次模擬測驗時唯一真正不知道會錯的題目,透過福音悔改的原則,我們可以將MySQL的知識再一次擴大,希望我們不會再錯了,各位觀眾可以參閱MySQL 5.7 Reference Manual :: 11 Data Types :: 11.2 Numeric Types :: 11.2.6 Out-of-Range and Overflow Handling或是簡體手冊MySQL 5.1参考手册 :: 11. 列类型::11.2. 数值类型。
繁體轉譯如下:
MySQL支援所有標準SQL數值資料型態。這些型態包括嚴格數值資料型態(INTEGER、SMALLINT、DECIMAL和NUMERIC),以及近似數值資料型態(FLOAT、REAL和DOUBLE PRECISION)。關鍵字INT是INTEGER的同義詞,關鍵字DEC是DECIMAL的同義詞。
BIT資料型態保存位元值,並且支援MyISAM、MEMORY、InnoDB和BDB資料表。
作為SQL標準的延伸,MySQL也支援整數型態TINYINT、MEDIUMINT和BIGINT。下面的資料表顯示了需要的每個整數型態的儲存和範圍。
型態 | 字節 | 最小值 | 最大值 |
(帶符號的/無符號的) | (帶符號的/無符號的) | ||
TINYINT | 1 | -128 | 127 |
0 | 255 | ||
SMALLINT | 2 | -32768 | 32767 |
0 | 65535 | ||
MEDIUMINT | 3 | -8388608 | 8388607 |
0 | 16777215 | ||
INT | 4 | -2147483648 | 2147483647 |
0 | 4294967295 | ||
BIGINT | 8 | -9223372036854775808 | 9223372036854775807 |
0 | 18446744073709551615 |
MySQL還支援選擇在該型態關鍵字後面的括號內指定整數值的顯示寬度(例如,INT(4))。該可選顯示寬度規定用於顯示寬度小於指定的欄位寬度的值時從左側填滿寬度。
顯示寬度並不限制可以在欄位內保存的值的範圍,也不限制超過欄位的指定寬度的值的顯示。
當結合可選延伸屬性ZEROFILL使用時, 預設補充的空格用零代替。例如,對於聲明為INT(5) ZEROFILL的欄位,值4檢索為00004。請注意如果在整數欄位保存超過顯示寬度的一個值,當MySQL為複雜聯接生成臨時資料表時會遇到問題,因為在這些情況下MySQL相信資料適合原欄位寬度。
所有整數型態可以有一個可選(非標準)屬性UNSIGNED。當您想要在欄位內只允許非負數和該欄位需要較大的上限數值範圍時可以使用無符號值。
浮點和定點類型也可以為UNSIGNED。同數類型,該屬性防止負值保存到欄位中。然而,與整數類型不同的是,欄位值的上範圍保持不變。
如果為一個數值欄位指定ZEROFILL,MySQL自動為該欄位新增UNSIGNED屬性。
對於浮點欄位型態,在MySQL中單精度值使用4個字節,雙精度值使用8個字節。
FLOAT型態用於資料表示近似數值資料型態。SQL標準允許在關鍵字FLOAT後面的括號內選擇用位指定精度(但不能為指數範圍)。MySQL還支援可選的只用於確定儲存大小的精度規定。0到23的精度對應FLOAT欄位的4字節單精度。24到53的精度對應DOUBLE欄位的8字節雙精度。
MySQL允許使用非標準語法:FLOAT(M,D)或REAL(M,D)或DOUBLE PRECISION(M,D)。這裡,「(M,D)」資料表示該值一共顯示M位整數,其中D位位於小數點後面。例如,定義為FLOAT(7,4)的一個欄位可以顯示為-999.9999。MySQL保存值時進行四捨五入,因此如果在FLOAT(7,4)欄位內插入999.00009,近似結果是999.0001。
MySQL將DOUBLE視為DOUBLE PRECISION(非標準延伸)的同義詞。MySQL還將REAL視為DOUBLE PRECISION(非標準延伸)的同義詞,除非SQL伺服器模式包括REAL_AS_FLOAT選項。
為了保證最大可能的可移植性,需要使用近似數值資料值儲存的代碼應使用FLOAT或DOUBLE PRECISION,不規定精度或位數。
DECIMAL和NUMERIC型態在MySQL中視為相同的類型。它們用於保存必須為確切精度的值,例如貨幣數據。當聲明該類型的列時,可以(並且通常要)指定精度和標度;例如:
salary DECIMAL(5,2)
在該例子中,5是精度,2是標度。精度資料表示保存值的主要位數,標度資料表示小數點後面可以保存的位數。
在MySQL 5.1中以二進制格式保存DECIMAL和NUMERIC值。
標準SQL要求salary欄位能夠用5位整數位和兩位小數保存任何值。因此,在這種情況下可以保存在salary欄位的值的範圍是從-999.99到999.99。
在標準SQL中,語法DECIMAL(M)等價於DECIMAL(M,0)。同樣,語法DECIMAL等價於DECIMAL(M,0),可以通過計算確定M的值。在MySQL 5.1中支援DECIMAL和NUMERIC資料型態的變數形式。M預設值是10。
DECIMAL或NUMERIC的最大位數是65,但具體的DECIMAL或NUMERIC欄位的實際範圍受具體欄位的精度或標度約束。如果此類欄位分配的值小數點後面的位數超過指定的標度允許的範圍,值被轉換為該標度。(具體操作與作業系統有關,但一般結果均被截取到允許的位數)。
BIT資料型態可用來保存位元值。BIT(M)型態允許儲存M位值。M範圍為1到64。
要指定位值,可以使用b’value‘符。value是一個用0和1編寫的二進制值。例如,b’111′和b’100000000′分別資料表示7和128。參見9.1.5節,「位元值」。
如果為BIT(M)欄位分配的值的長度小於M位,在值的左邊用0填充。例如,為BIT(6)列分配一個值b’101′,其效果與分配b’000101′相同。
當要在一個數值欄位內保存一個超出該欄位允許範圍的值時,MySQL的操作取決於此時有效的SQL模式。如果模式未設置,MySQL將值裁剪到範圍的相應端點,並保存裁減好的值。但是,如果模式設置為traditional(「嚴格模式」),超出範圍的值將被拒絕並提示錯誤,並且根據SQL標準插入會失敗。參見5.3.2節,「SQL伺服器模式」。
如果INT欄位是UNSIGNED,欄位範圍的大小相同,但其端點會變為到0和4294967295。如果您試圖保存-9999999999和9999999999,以非嚴格模式保存到欄位中的值是0和4294967296。
如果在浮點或定點列中分配的值超過指定(或預設)精度和標度規定的範圍,MySQL以非嚴格模式保存資料表示範圍相應端點的值。
當MySQL沒有工作在嚴格模式時,對於ALTER TABLE、LOAD DATA INFILE、UPDATE和多行INSERT語句,由於裁剪髮生的轉換將報告為警告。當MySQL工作在嚴格模式時,這些語句將失敗,並且部分或全部值不會插入或更改,取決於是否資料表為事務資料表和其它因素。詳情參見5.3.2節,「SQL伺服器模式」。
這裡的錯誤就是1264的錯誤,由於中文手冊比較沒有詳述其錯誤,所以看原文的手冊可以比較清楚,請我們一起來牢記這樣的題型。
2 則留言