Проверьте, является ли varchar числом (TSQL) - PullRequest
55 голосов
/ 05 января 2011

Есть ли простой способ выяснить, является ли varchar числом?

Примеры:

abc123 -> без номера

123 -> да, это число

Спасибо:)

Ответы [ 9 ]

128 голосов
/ 05 января 2011

ISNUMERIC не подойдет - он сообщает, что строку можно преобразовать в любой числовых типов, что почти всегда является бессмысленной информацией. Например, все следующие являются числовыми, согласно ISNUMERIC:

£, $, 0d0

Если вы хотите проверить цифры и только цифры, то вам нужно отрицательное LIKE-выражение:

not Value like '%[^0-9]%'
29 голосов
/ 05 января 2011

ISNUMERIC сделает

Проверьте также раздел NOTES в статье.

26 голосов
/ 05 января 2011

вы можете проверить вот так

declare @vchar varchar(50)
set @vchar ='34343';
select case when @vchar not like '%[^0-9]%' then 'Number' else 'Not a Number' end
17 голосов
/ 18 марта 2016

Используя SQL Server 2012+, вы можете использовать функции TRY_ *, если у вас есть особые потребности. Например,

-- will fail for decimal values, but allow negative values
TRY_CAST(@value AS INT) IS NOT NULL 

-- will fail for non-positive integers; can be used with other examples below as well, or reversed if only negative desired
TRY_CAST(@value AS INT) > 0

-- will fail if a $ is used, but allow decimals to the specified precision
TRY_CAST(@value AS DECIMAL(10,2)) IS NOT NULL 

-- will allow valid currency
TRY_CAST(@value AS MONEY) IS NOT NULL  

-- will allow scientific notation to be used like 1.7E+3
TRY_CAST(@value AS FLOAT) IS NOT NULL 
8 голосов
/ 01 декабря 2014

Я столкнулся с необходимостью разрешить десятичные значения, поэтому я использовал not Value like '%[^0-9.]%'

3 голосов
/ 14 ноября 2015

Ответ Wade73 на десятичные дроби не совсем работает. Я изменил его, чтобы разрешить только одну десятичную точку.

declare @MyTable table(MyVar nvarchar(10));
insert into @MyTable (MyVar) 
values 
    (N'1234')
    , (N'000005')
    , (N'1,000')
    , (N'293.8457')
    , (N'x')
    , (N'+')
    , (N'293.8457.')
    , (N'......');

-- This shows that Wade73's answer allows some non-numeric values to slip through.
select * from (
    select 
        MyVar
        , case when MyVar not like N'%[^0-9.]%' then 1 else 0 end as IsNumber 
    from 
        @MyTable
) t order by IsNumber;

-- Notice the addition of "and MyVar not like N'%.%.%'".
select * from (
    select 
        MyVar
        , case when MyVar not like N'%[^0-9.]%' and MyVar not like N'%.%.%' then 1 else 0 end as IsNumber 
    from 
        @MyTable
) t 
order by IsNumber;
1 голос
/ 09 февраля 2017

Damien_The_Unbeliever отметил, что он был хорош только для цифр

Wade73 добавил немного для обработки десятичных точек

neizan сделал дополнительный твик, как нигде не было

К сожалению, ни одна из них не обрабатывает отрицательные значения, и у них, похоже, есть проблемы с запятой в значении ...

Вот мой твик для подбора отрицательных значений и запятых

declare @MyTable table(MyVar nvarchar(10));
insert into @MyTable (MyVar) 
values 
(N'1234')
, (N'000005')
, (N'1,000')
, (N'293.8457')
, (N'x')
, (N'+')
, (N'293.8457.')
, (N'......')
, (N'.')
, (N'-375.4')
, (N'-00003')
, (N'-2,000')
, (N'3-3')
, (N'3000-')
;

-- This shows that Neizan's answer allows "." to slip through.
select * from (
select 
    MyVar
    , case when MyVar not like N'%[^0-9.]%' then 1 else 0 end as IsNumber 
from 
    @MyTable
) t order by IsNumber;

-- Notice the addition of "and MyVar not like '.'".
select * from (
select 
    MyVar
    , case when MyVar not like N'%[^0-9.]%' and MyVar not like N'%.%.%' and MyVar not like '.' then 1 else 0 end as IsNumber 
from 
    @MyTable
) t 
order by IsNumber;

--Trying to tweak for negative values and the comma
--Modified when comparison
select * from (
select 
    MyVar
    , case 
        when MyVar not like N'%[^0-9.,-]%' and MyVar not like '.' and isnumeric(MyVar) = 1 then 1
        else 0 
    end as IsNumber 
from 
    @MyTable
) t 
order by IsNumber;
1 голос
/ 17 февраля 2016

Код Neizan позволяет значениям только "."через.Риск стать слишком педантичным, я добавил еще одно предложение AND.

declare @MyTable table(MyVar nvarchar(10));
insert into @MyTable (MyVar) 
values 
    (N'1234')
    , (N'000005')
    , (N'1,000')
    , (N'293.8457')
    , (N'x')
    , (N'+')
    , (N'293.8457.')
    , (N'......')
    , (N'.')
    ;

-- This shows that Neizan's answer allows "." to slip through.
select * from (
    select 
        MyVar
        , case when MyVar not like N'%[^0-9.]%' then 1 else 0 end as IsNumber 
    from 
        @MyTable
) t order by IsNumber;

-- Notice the addition of "and MyVar not like '.'".
select * from (
    select 
        MyVar
        , case when MyVar not like N'%[^0-9.]%' and MyVar not like N'%.%.%' and MyVar not like '.' then 1 else 0 end as IsNumber 
    from 
        @MyTable
) t 
order by IsNumber;
1 голос
/ 03 ноября 2015
DECLARE @A nvarchar(100) = '12'
IF(ISNUMERIC(@A) = 1)
BEGIN
    PRINT 'YES NUMERIC'
END
...