本章主要介绍我们编程时遇到的一些常见 javascript 错误。
1. 意外使用赋值运算符
如果程序员在 if 语句中意外使用赋值运算符(=)而不是比较运算符(===),javascript 程序可能会产生一些无法预料的结果。
这条 if 语句返回 false(正如预期),因为 x 不等于 10:
var x = 0; if (x == 10)
这条 if 语句返回 true(也许不像预期),因为 10 为 true:
var x = 0; if (x = 10)
这条 if 语句返回 false(也许不像预期),因为 0 为 false:
var x = 0; if (x = 0)
赋值总是返回赋值的值。
2. 期望松散的比较
在常规比较中,数据类型不重要。这条 if 语句返回 true:
var x = 10; var y = "10"; if (x == y)
在严格比较中,数据类型确实重要。这条 if 语句返回 false:
var x = 10; var y = "10"; if (x === y)
有一个常见的错误是忘记在 switch 语句中使用严格比较:
这条 switch 语句会显示提示框:
var x = 10; switch(x) { case 10: alert("hello"); }
这条 switch 语句不会显示提示框:
var x = 10; switch(x) { case "10": alert("hello"); }
3. 令人困惑的加法和级联
加法用于加数值。
级联(concatenation)用于加字符串。
在 javascript 中,这两种运算均使用相同的 + 运算符。
正因如此,将数字作为数值相加,与将数字作为字符串相加,将产生不同的结果:
var x = 10 + 5; // x 中的结果是 15 var x = 10 + "5"; // x 中的结果是 "105"
如果是两个变量相加,很难预测结果:
var x = 10; var y = 5; var z = x + y; // z 中的结果是 15 var x = 10; var y = "5"; var z = x + y; // z 中的结果是 "105"
4. 令人误解的浮点
javascript 中的数字均保存为 64 位的浮点数(floats)。
所有编程语言,包括 javascript,都存在处理浮点值的困难:
var x = 0.1;
var y = 0.2;
var z = x + y // z 中的结果并不是 0.3
为了解决上面的问题,请使用乘除运算:
范例
var z = (x * 10 + y * 10) / 10; // z 中的结果将是 0.3
5. 对 javascript 字符串换行
javascript 允许您把一条语句换行为两行:
例子 1
var x = "hello world!";
但是,在字符串中间来换行是不对的:
例子 2
var x = "hello world!";
如果必须在字符串中换行,则必须使用反斜杠:
例子 3
var x = "hello \ world!";
6. 错位的分号
因为一个错误的分号,此代码块无论 x 的值如何都会执行:
if (x == 19);
{
// code block }
7. 对 return 语句进行换行
在一行的结尾自动关闭语句是默认的 javascript 行为。
正因如此,下面两个例子返回相同的结果:
例子 1
function myfunction(a) { var power = 10 return a * power }
例子 2
function myfunction(a) { var power = 10; return a * power; }
javascript 也允许您将一条语句换行为两行。
正因如此,例子 3 也将返回相同的结果:
例子 3
function myfunction(a) { var power = 10; return a * power; }
但是,如果把 return 语句换行为两行会发生什么呢:
例子 4
function myfunction(a) { var power = 10; return a * power; }
此函数将返回 undefined!
为什么呢?因为 javascript 认为你的意思是:
例子 5
function myfunction(a) { var power = 10; return; a * power; }
8. 解释
如果某条语句是不完整的:
var
javascript 将通过读取下一行来完成这条语句:
power = 10;
但是由于这条语句是完整的:
return
javascript 会自动关闭该语句:
return;
发生这种情况是因为,在 javascript 中,用分号来关闭(结束)语句是可选的。
javascript 会在行末关闭 return 语句,因为它本身就是一条完整的语句。
所以,绝不要对 return 语句进行换行。
9. 通过命名索引来访问数组
很多编程语言支持带有命名索引的数组。
带有命名索引的数组被称为关联数组(或散列)。
javascript 不支持带有命名索引的数组。
在 javascript 中,数组使用数字索引:
范例
var person = []; person[0] = "bill"; person[1] = "gates"; person[2] = 46; var x = person.length; // person.length 将返回 3 var y = person[0]; // person[0] 将返回 "john"
在 javascript 中,对象使用命名索引。
如果您使用命名索引,那么在访问数组时,javascript 会将数组重新定义为标准对象。
在自动重定义之后,数组方法或属性将产生未定义或非正确的结果:
范例
var person = []; person["firstname"] = "bill"; person["lastname"] = "gates"; person["age"] = 46; var x = person.length; // person.length 将返回 0 var y = person[0]; // person[0] 将返回 undefined
10. 用逗号来结束定义
对象和数组定义中的尾随逗号在 ecmascript 5 中是合法的。
对象范例:
person = {firstname:"bill", lastname:"gates", age:62,}
数组范例:
points = [35, 450, 2, 7, 30, 16,];
警告!!
internet explorer 8 会崩溃。
json 不允许尾随逗号。
json:
person = {firstname:"bill", lastname:"gates", age:62}
json:
points = [35, 450, 2, 7, 30, 16];
11. undefined 不是 null
javascript 对象、变量、属性和方法可以是未定义的。
此外,空的 javascript 对象的值可以为 null。
这可能会使测试对象是否为空变得有点困难。
您可以通过测试类型是否为 undefined,来测试对象是否存在:
范例
if (typeof myobj === "undefined")
但是您无法测试对象是否为 null,因为如果对象未定义,将抛出错误:
不正确的:
if (myobj === null)
要解决此问题,必须测试对象是否为 null,而不是未定义。
但这仍然会引发错误:
不正确的:
if (myobj !== null && typeof myobj !== "undefined")
因此,在测试非 null 之前,必须先测试未定义:
正确的:
if (typeof myobj !== "undefined" && myobj !== null)
12. 期望块级范围
javascript 不会为每个代码块创建新的作用域。
很多编程语言都是如此,但是 javascript 并非如此。
认为这段代码会返回 undefined,是新的 javascript 开发者的常见错误:
范例
for (var i = 0; i < 10; i++) {
// 代码块 }
return i;