Why does isNaN(“ ”) (string with spaces) equal false?
在JavaScript中,为什么
我正在对文本输入字段执行数字运算,并且正在检查该字段是
JavaScript将一个空字符串解释为0,然后失败了isNAN测试。您可以先在字符串上使用parseInt,它将不会将空字符串转换为0。然后结果应为isNAN。
您可能会感到惊讶,也可能不会惊讶,但是这里有一些测试代码向您展示JavaScript引擎的古怪之处。
1 2 3 4 5 6 7 8 9 10 11 | document.write(isNaN("")) // false document.write(isNaN("")) // false document.write(isNaN(0)) // false document.write(isNaN(null)) // false document.write(isNaN(false)) // false document.write("" == false) // true document.write("" == 0) // true document.write("" == 0) // true document.write("" == false) // true document.write(0 == false) // true document.write("" =="") // false |
所以这意味着
1 | "" == 0 == false |
和
1 | "" == 0 == false |
但
1 | "" !="" |
玩得开心 :)
为了更好地理解它,请打开第43页" ToNumber应用于字符串类型"的Ecma-Script规范pdf。
如果字符串具有数字语法,可以包含任意数量的空格字符,则可以将其转换为Number类型。空字符串的求值为0。字符串" Infinity"也应为
1 | isNaN('Infinity'); // false |
尝试使用:
1 | alert(isNaN(parseInt(" "))); |
要么
1 | alert(isNaN(parseFloat(" "))); |
由于
When the argument to the isNaN function is not of type Number, the value is first coerced to a Number. The resulting value is then tested to determine whether it is NaN.
您可能需要检查以下全面的答案,其中涵盖了NaN比较是否相等。
How to test if a JavaScript variable is
NaN
我认为这是因为Javascript的类型:
1 2 | alert(' ' * 1); // 0 alert('x' * 1); // NaN |
不完全正确的答案
Antonio Haley对此高度评价和接受的答案做出了错误的假设,认为此过程通过JavaScript的
You can use parseInt on the string ... The result should then fail isNAN.
我们可以使用字符串
1 2 | parseInt("123abc") // 123 (a number... isNaN("123abc") // true ...which is not a number) |
这样,我们可以看到JavaScript的
正确答案
ECMAScript-262在第18.2.3节中定义
18.2.3
isNaN (Number)The
isNaN function is the%isNaN% intrinsic object. When theisNaN function is called with one argument number, the following steps are taken:Let num be ?ToNumber(number) .If num isNaN , returntrue .Otherwise, return false .
它引用的
问题中给出的第一个示例是一个字符串,其中只包含空格字符。本节指出:
A
StringNumericLiteral that is empty or contains only white space is converted to+0 .
因此,
同一部分还指出:
If the grammar cannot interpret the
String as an expansion ofStringNumericLiteral , then the result ofToNumber isNaN .
如果不引用该节中包含的所有检查,则问题中给出的
如果您想实现一个准确的isNumber函数,可以使用Javascript的一种方法:Douglas Crockford撰写的The Good Parts [第105页]
1 2 3 4 | var isNumber = function isNumber(value) { return typeof value === 'number' && isFinite(value); } |
函数
ECMAScript 3-5为typeof运算符定义了以下返回值:
- 未定义
- 对象(空,对象,数组)
- 布尔
- 数
- 串
- 功能
最好将我们的测试包装在一个函数体中:
1 2 3 4 5 6 | function isNumber (s) { return typeof s == 'number'? true : typeof s == 'string'? (s.trim() === ''? false : !isNaN(s)) : (typeof s).match(/object|function/)? false : !isNaN(s) } |
此函数并非旨在测试变量类型,而是测试强制值。例如,布尔值和字符串被强制转换为数字,因此也许您可能想将此函数称为
如果不需要测试除字符串和数字以外的其他类型,则以下代码段可用作某些条件的一部分:
1 2 | if (!isNaN(s) && s.toString().trim()!='') // 's' can be boolean, number or string alert("s is a number") |
我不确定为什么,但是要解决该问题,您可以始终在检查之前修剪空白。无论如何,您可能想要这样做。
如果您确实想要正确检查它是否为整数,建议您使用以下函数:
1 2 3 4 | function isInteger(s) { return Math.ceil(s) == Math.floor(s); } |
从MDN:
Since the very earliest versions of the
isNaN function specification, its behavior for non-numeric arguments has been confusing. When the argument to theisNaN function is not of type Number, the value is first coerced to a Number. The resulting value is then tested to determine whether it isNaN . Thus for non-numbers that when coerced to numeric type result in a valid non-NaN numeric value (notably the empty string and boolean primitives, which when coerced give numeric values zero or one), the"false" returned value may be unexpected; the empty string, for example, is surely"not a number."
还请注意,对于ECMAScript 6,现在还有
In comparison to the global
isNaN() function,Number.isNaN() doesn't suffer the problem of forcefully converting the parameter to a number. This means it is now safe to pass values that would normally convert toNaN , but aren't actually the same value asNaN . This also means that only values of the type number, that are alsoNaN , returntrue .
不幸:
如博客文章中概述的那样,甚至ECMAScript 6
顺便说一句。这对于所有内置函数都是常见的-如果它们期望某个类型的参数,则将使用标准转换函数来转换实际参数。所有基本类型(布尔型,字符串,数字,对象,日期,空,未定义)之间都有标准转换。
从
-
Number(" ") 计算为0 -
Number(" x") 计算为NaN
鉴于此,
真正的问题是为什么标准的字符串到数字的转换像它那样工作。字符串到数字的转换实际上是为了将数字字符串(例如" 123"或" 17.5e4")转换为等效数字。转换首先跳过初始空格(因此" 123"有效),然后尝试将其余部分解析为数字。如果它不可解析为数字(不是" x"),则结果为NaN。但是有一个明确的特殊规则,即将空或仅空白的字符串转换为0。因此,可以解释该转换。
参考:http://www.ecma-international.org/ecma-262/5.1/#sec-9.3.1
如其他解释所述,
但是,我发现
此检查将适用于正数,负数和带小数点的数字,因此我认为它可以涵盖所有常见的数值情况。
当检查带有空格或
if (value.match(/\s/) || isNaN(value)) {
// do something
}
此功能似乎在我的测试中有效
1 2 3 4 5 6 7 8 9 10 11 12 | function isNumber(s) { if (s ==="" || s === null) { return false; } else { var number = parseInt(s); if (number == 'NaN') { return false; } else { return true; } } } |
我写了这个快速的小功能来帮助解决这个问题。
1 2 3 | function isNumber(val) { return (val != undefined && val != null && val.toString().length > 0 && val.toString().match(/[^0-9\.\-]/g) == null); }; |
它只是检查所有不是数字(0-9),不是'-'或'。',不是undefined,null或empty的字符,如果没有匹配项,则返回true。 :)
JavaScript内置的isNaN函数-默认情况下应该是-"动态类型运算符"。
因此(在DTC过程中)所有值都可能产生一个简单的true |。 false,例如
这意味着提供的参数将首先进行如下转换:
1 2 3 4 | function isNaNDemo(arg){ var x = new Number(arg).valueOf(); return x != x; } |
说明:
在函数主体的顶行,我们(首先)试图将自变量成功转换为数字对象。 (第二)使用点运算符-为了我们自己的方便-我们立即剥离所创建对象的原始值。
在第二行中,我们采用在上一步中获得的值,以及NaN不等于宇宙中任何事物,甚至不等于它本身的事实的优势,例如:
这样,仅当且仅当提供的参数返回尝试转换为数字对象(即非数字)的失败尝试时,函数返回才会产生true。例如NaN。
isNaNstatic()
但是,对于静态类型运算符-如果需要和需要,我们可以编写一个简单得多的函数,例如:
1 2 3 | function isNaNstatic(x){ return x != x; } |
并完全避免使用DTC,以便如果该参数不是明确的NaN数字,则它将返回false。因此,请针对以下各项进行测试:
然而:
但是与
P.S:
isNaN的静态版本在现代编码方案中可能非常有用。这很可能是我花时间张贴此文章的主要原因之一。
问候。
也就是说,isNaN(foo)等同于isNaN(Number(foo))
出于显而易见的原因,它接受具有所有数字作为数字的任何字符串。对于前。
1 2 3 4 5 6 7 8 9 10 11 12 13 | isNaN(123) //false isNaN(-1.23) //false isNaN(5-2) //false isNaN(0) //false isNaN('123') //false isNaN('Hello') //true isNaN('2005/12/12') //true isNaN('') //false isNaN(true) //false isNaN(undefined) //true isNaN('NaN') //true isNaN(NaN) //true isNaN(0 / 0) //true |
我用这个
1 2 3 4 5 6 7 8 9 10 11 12 | function isNotANumeric(val) { if(val.trim && val.trim() =="") { return true; } else { return isNaN(parseFloat(val * 1)); } } alert(isNotANumeric("100")); // false alert(isNotANumeric("1a")); // true alert(isNotANumeric("")); // true alert(isNotANumeric(" ")); // true |
这是ECMAScript中isNaN()的定义
1 2 3 4 | 1. Let num be ToNumber(number). 2. ReturnIfAbrupt(num). 3. If num is NaN, return true. 4. Otherwise, return false. |
尝试将任何值转换为Number。
1 2 3 | Number("") // 0 Number("x") // NaN Number(null) // 0 |
如果要确定值是否为
关于什么
1 2 3 4 5 | function isNumberRegex(value) { var pattern = /^[-+]?\d*\.?\d*$/i; var match = value.match(pattern); return value.length > 0 && match != null; } |