3

我将一个平面文件导入 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,但是当我尝试更改字段时仍然会出现错误。有谁知道还有什么可能导致此错误?

4

2 回答 2

3

所以,ISNUMERIC()远非完美e对于, $, '€' 和tab, 以及许多其他你不会想到的值,它可以返回 true 。但是当然,您不能将它们转换为每种数字类型(ISNUMERIC()仅标识它可以转换为至少一种数字类型)。LTRIM()并且RTRIM()不会帮助,因为他们不会清除tab和其他类似的字符。

要识别实际上不是数字的行:

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 

一旦您确定了这些行并修复了数据,请,请,请也修复表格。

于 2013-08-09T22:08:32.073 回答
2

您可以使用此查询找到有问题的行:

SELECT *
FROM MAWB
WHERE TRY_CONVERT(decimal(12,2),Weight) IS NULL
AND Weight IS NOT NULL
于 2013-08-10T02:26:24.067 回答