Как мне узнать, имеет ли переменная числовое значение в Perl?

76

Есть ли простой способ в Perl, который позволит мне определить, является ли данная переменная числовой? Что-то вроде:

if (is_number($x))
{ ... }

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

Теги:
numbers

15 ответов

120
Лучший ответ

Используйте Scalar::Util::looks_like_number() который использует внутреннюю функцию Perl C API look_like_number(), что, вероятно, является наиболее эффективным способом для этого. Обратите внимание, что строки "inf" и "бесконечность" рассматриваются как числа.

Пример:

#!/usr/bin/perl

use warnings;
use strict;

use Scalar::Util qw(looks_like_number);

my @exprs = qw(1 5.25 0.001 1.3e8 foo bar 1dd inf infinity);

foreach my $expr (@exprs) {
    print "$expr is", looks_like_number($expr) ? '' : ' not', " a number\n";
}

Дает этот результат:

1 is a number
5.25 is a number
0.001 is a number
1.3e8 is a number
foo is not a number
bar is not a number
1dd is not a number
inf is a number
infinity is a number

Смотрите также:

  • 1
    И, как обычно с perl docs, найти фактическое определение того, что делает функция, довольно сложно. Следуя perldoc perlapi говорит нам: Проверьте, выглядит ли содержимое SV как число (или число). «Inf» и «Infinity» обрабатываются как числа (поэтому не будут выдавать нечисловое предупреждение), даже если ваш atof () их не подставляет. Вряд ли тестируемая спецификация ...
  • 2
    С описанием в Scalar :: Util все в порядке, look_like_number сообщает вам, является ли ваш ввод чем-то, что Perl будет рассматривать как число, что не обязательно является лучшим ответом на этот вопрос. Упоминание atof не имеет значения, atof не является частью CORE :: или POSIX (вы должны смотреть на strtod, который относится к atof и / is / part of POSIX) и предполагать, что Perl считает число допустимым числовым вводом к функциям C, очевидно, очень неправильно.
Показать ещё 4 комментария
24

Проверьте модуль CPAN Regexp:: Common. Я думаю, что он делает именно то, что вам нужно, и обрабатывает все крайние случаи (например, реальные числа, научную нотацию и т.д.). например.

use Regexp::Common;
if ($var =~ /$RE{num}{real}/) { print q{a number}; }
21

Первоначальный вопрос заключался в том, как определить, является ли переменная числовой, а не если она имеет числовое значение.

Существует несколько операторов, которые имеют отдельные режимы работы для числовых и строковых операндов, где "числовой" означает все, что изначально было числом или когда-либо использовалось в числовом контексте (например, в $x = "123"; 0+$x, перед добавлением, $x является строкой, затем считается числовой).

Один из способов сказать следующее:

if ( length( do { no warnings "numeric"; $x & "" } ) ) {
    print "$x is numeric\n";
}
  • 0
    Отлично, спасибо! Это именно то, что я искал.
  • 0
    Если я упаковываю вашу подпрограмму в подпрограмму, я получаю странное поведение в том, что она правильно определяет нечисловые значения, пока не опробую первое числовое значение, которое также правильно определено как истинное, но затем все остальное оттуда тоже верно. Когда я помещаю eval вокруг части length (...), она всегда работает нормально. Есть идеи, что мне не хватало? sub numeric { $obj = shift; no warnings "numeric"; return eval('length($obj & "")'); }
Показать ещё 4 комментария
8

Простой (и, может быть, упрощенный) ответ на вопрос состоит в содержании $x числового:

if ($x  eq  $x+0) { .... }

Он выполняет текстовое сравнение исходного $x с $x, преобразованным в числовое значение.

  • 0
    Это будет выдавать предупреждения, если вы используете «-w» или «использовать предупреждения;».
  • 0
    Предупреждения могут быть удалены $x eq (($x+0)."") Однако хуже проблема в том, что под этой функцией "1.0" не является числовым
Показать ещё 1 комментарий
7

Обычно проверка числа выполняется с помощью регулярных выражений. Этот код будет определять, является ли что-то числовое, а также проверять переменные undefined, чтобы не выдавать предупреждения:

sub is_integer {
   defined $_[0] && $_[0] =~ /^[+-]?\d+$/;
}

sub is_float {
   defined $_[0] && $_[0] =~ /^[+-]?\d+(\.\d+)?$/;
}

Вот некоторые материалы для чтения, на которые вы должны посмотреть.

  • 2
    Это пропускает много случаев, в том числе научных обозначений.
  • 2
    Я думаю, что это немного отвлекает, особенно когда спрашивающий сказал / просто /. Многие случаи, в том числе научные записи, вряд ли просты. Если бы не использовать это для модуля, я бы не стал беспокоиться о таких деталях. Иногда простота лучше. Не кладите шоколадный сироп в корову, чтобы сделать шоколадное молоко!
Показать ещё 2 комментария
3

Я не верю, что для этого есть что-то построенное. Больше, чем вы хотели видеть по этому вопросу, см. Perlmonks при обнаружении числа

3

Не идеально, но вы можете использовать регулярное выражение:

sub isnumber 
{
    shift =~ /^-?\d+\.?\d*$/;
}
  • 0
    Та же проблема, что и в ответе andrewrk: пропускает много даже простых случаев, например, «.7»
2

rexep не идеально... это:

use Try::Tiny;

sub is_numeric {
  my ($x) = @_;
  my $numeric = 1;
  try {
    use warnings FATAL => qw/numeric/;
    0 + $x;
  }
  catch {
    $numeric = 0;
  };
  return $numeric;
}
2

Несколько более надежное регулярное выражение можно найти в Regexp:: Common.

Похоже, вы хотите знать, считает ли Perl переменную числовой. Здесь функция, которая прерывает это предупреждение:

sub is_number{
  my $n = shift;
  my $ret = 1;
  $SIG{"__WARN__"} = sub {$ret = 0};
  eval { my $x = $n + 1 };
  return $ret
}

Другой вариант - отключить предупреждение локально:

{
  no warnings "numeric"; # Ignore "isn't numeric" warning
  ...                    # Use a variable that might not be numeric
}

Обратите внимание, что нечисловые переменные будут безшовно преобразованы в 0, что, вероятно, все равно вам нужно.

1

Я нашел это интересным, хотя

if ( $value + 0 eq $value) {
    # A number
    push @args, $value;
} else {
    # A string
    push @args, "'$value'";
}
  • 0
    Вы должны объяснить лучше, вы говорите, что вы находите это интересным, но отвечает ли он оператору? Постарайтесь объяснить, почему ваш ответ является решением вопроса
  • 0
    Например, мой $ value равен 1, $ value + 0 остается тем же 1. При сравнении с $ value 1 равен 1. Если $ value является строкой, скажем «swadhi», то $ value + 0 становится ascii значением строки «swadhi» + 0 = какой-то другой номер.
1

Попробуйте следующее:

If (($x !~ /\D/) && ($x ne "")) { ... }
0

Вы можете использовать регулярные выражения, чтобы определить, является ли $foo числом (или нет).

Взгляните сюда: Как определить, является ли скаляр числом?

0

Лично я считаю, что путь - это полагаться на внутренний контекст Perl, чтобы сделать решение пуленепробиваемым. Хорошее регулярное выражение может соответствовать всем допустимым числовым значениям и ни одному из нечисловых (или наоборот), но поскольку существует способ использования той же логики, которую использует интерпретатор, должно быть безопаснее полагаться на это непосредственно.

Поскольку я, как правило, запускаю свои сценарии с помощью -w, мне пришлось объединить идею сравнения результата "значение плюс ноль" с исходным значением с помощью подхода no warnings на основе @ysth:

do { 
    no warnings "numeric";
    if ($x + 0 ne $x) { return "not numeric"; } else { return "numeric"; }
}
-1

if (определяется $x && $x! ~ m/\ D/) {} или $ x = 0, если! $Х; if ($ x! ~ m/\ D/) {}

Это небольшая вариация на ответе Викея, но позвольте мне объяснить мои аргументы в пользу изменения.

Выполнение регулярного выражения по значению undefined вызовет ошибку из-за ошибки и приведет к выходу кода во многих, если не в большинстве средах. Тестирование, если значение определено или задание по умолчанию, как я сделал в альтернативном примере перед запуском выражения, как минимум, сохранит ваш журнал ошибок.

-2

Эта функция работает для меня:

    sub IS_Integer() 
    {
        my $Text = shift;
        my $Integer = 0;

        if ($Text =~ /\D/)
        {
            $Integer = 1;
        }
        if ($Text =~ /^\d+$/)
        { 
            $Integer = 1;
        }
        if ($Text =~ /^-?\d+$/)       
        { 
            $Integer = 1;
        }
        if ($Text =~ /^[+-]?\d+$/)    
        { 
            $Integer = 1; 
        }
        if ($Text =~ /^-?\d+\.?\d*$/) 
        {
            $Integer = 1; 
        }
        if ($Text =~ /^-?(?:\d+(?:\.\d*)?&\.\d+)$/) 
        {
            $Integer = 1;
        }
        if ($Text =~ /^([+-]?)(?=\d&\.\d)\d*(\.\d*)?([Ee]([+-]?\d+))?$/)
        {
            $Integer = 1;
        }

        return $Integer;
    }

Ещё вопросы

Сообщество Overcoder
Наверх
Меню