我将一个平面文件导入SQL数据库,并将所有字段创建为varchar(50)。我能够更改大多数字段的数据类型,但在权重字段中出现错误。权重字段中的所有数据总长度小于6个字符,并且可以是整数或小数。我试过了两个:
UPDATE MAWB
SET Weight = CAST(Weight AS decimal(12,2))
和
ALTER TABLE MAWB
ALTER COLUMN [Weight] decimal(10,2)
我总是收到错误:将数据类型varchar转换为数字时出错。
我已经检查过所有字段都被视为数字:
SELECT COUNT(*)
FROM MAWB
WHERE ISNUMERIC(Weight) = 0
我也尝试过LTRIM和RTRIM只是为了安全,但在我尝试更改字段时仍然会出错。有谁知道还有什么可能导致这个错误?
答案 0 :(得分:3)
所以,ISNUMERIC()
is far from perfect。对于e
,$
,'€'和 tab 等值,以及其他许多您不希望的值,它可以返回true。但是当然你不能将它们转换为每种数字类型(ISNUMERIC()
只是标识它可以转换为至少一种数字类型)。 LTRIM()
和RTRIM()
无效,因为他们无法清除标签和其他类似字符。
识别真正不是数字的行:
SELECT weight FROM dbo.table WHERE column NOT LIKE '%[^0-9.-]%';
或者你可以写一个更精细的标量函数,比如上面链接中的那个,它可以处理更多的边缘情况:
CREATE FUNCTION dbo.isReallyNumeric
(
@num VARCHAR(64)
)
RETURNS BIT
BEGIN
IF LEFT(@num, 1) = '-'
SET @num = SUBSTRING(@num, 2, LEN(@num))
DECLARE @pos TINYINT
SET @pos = 1 + LEN(@num) - CHARINDEX('.', REVERSE(@num))
RETURN CASE
WHEN PATINDEX('%[^0-9.-]%', @num) = 0
AND @num NOT IN ('.', '-', '+', '^')
AND LEN(@num)>0
AND @num NOT LIKE '%-%'
AND
(
((@pos = LEN(@num)+1)
OR @pos = CHARINDEX('.', @num))
)
THEN
1
ELSE
0
END
END
GO
一旦您确定了这些行并修复了数据,请另外修改表格。
答案 1 :(得分:2)
您可以使用此查询找到有问题的行:
SELECT *
FROM MAWB
WHERE TRY_CONVERT(decimal(12,2),Weight) IS NULL
AND Weight IS NOT NULL