try-catch
语句,作为 javascript 中处理异常的一种标准方式。基本的语法如下所示:
try{
// 可能会导致错误的代码
} catch(error){
// 在错误发生时怎么处理
}
如果 try
块中的任何代码发生了错误,就会立即退出代码执行过程,然后接着执行 catch
块。此时,catch
块会接收到一个包含错误信息的对象。即使你不想使用这个错误对象,也要给它起个名字。这个对象中包含一个保存着错误消息的 message
属性和一个保存错误类型的 name
属性。
finally 子句
虽然在 try-catch
语句中是可选的,但 finally
子句一经使用,其代码无论如何都会执行。换句话说,try
语句块中的代码全部正常执行,finally
子句会执行;如果因为出错而执行了 catch
语句块,finally
子句照样还会执行。只要代码中包含 finally
子句,则无论 try
或 catch
语句块中包含什么代码——甚至 return
语句,都不会阻止 finally
子句的执行。来看下面这个函数。
function testfinally(){
try {
return 2;
} catch (error){
return 1;
} finally {
return 0;
}
}
这个函数在 try-catch
语句的每一部分都放了一条 return
语句。表面上看,调用这个函数会返回 2
,因为返回 2
的 return
语句位于 try
语句块中,而执行该语句又不会出错。可是,由于最后还有一个 finally
子句,结果就会导致该 return
语句被忽略;也就是说,调用这个函数只能返回 0
。如果把 finally
子句拿掉,这个函数将返回 2
。
只要代码中包含
finally
子句,那么无论try
还是catch
语句块中的return
语句都将被忽略。
如果提供 finally
子句,则 catch
子句就成了可选的(catch
或 finally
有一个即可)。
错误类型
执行代码期间可能会发生的错误有多种类型。每种错误都有对应的错误类型,而当错误发生时,就会抛出相应类型的错误对象。js有下列 7 种错误类型:
- error
- evalerror
- rangeerror
- referenceerror
- syntaxerror
- typeerror
- urierror
其中,error 是基类型,其他错误类型都继承自该类型。因此,所有错误类型共享了一组相同的属性(错误对象中的方法全是默认的对象方法)。error 类型的错误很少见,如果有也是浏览器抛出的;这个基类型的主要目的是供开发人员抛出自定义错误。
evalerror 类型的错误会在使用 eval()
函数而发生异常时被抛出。如果以非直接调用的方式使用 eval
属性的值,或者为 eval
属性赋值,简单地说,如果没有把 eval()
当成函数调用,就会抛出错误,例如:
new eval(); //抛出 evalerror
eval = foo; //抛出 evalerror
在实践中,浏览器不一定会在应该抛出错误时就抛出 evalerror。例如,firefox 4 和 ie8 对第一种情况会抛出 typeerror,而第二种情况会成功执行,不发生错误。
rangeerror 类型的错误会在数值超出相应范围时触发。例如,在定义数组时,如果指定了数组不支持的项数(如-20
或 number.max_value
),就会触发这种错误。
referenceerror 类型的错误会在访问不存在的变量时发生。
syntaxerror 类型的错误会在我们把语法错误的 javascript 字符串传入 eval()
函数时,就会导致此类错误。如果语法错误的代码出现在 eval()
函数之外,则不太可能使用 syntaxerror,因为此时的语法错误会导致 javascript 代码立即停止执行。
typeerror 类型在 javascript 中会经常用到,在变量中保存着意外的类型时,或者在访问不存在的方法时,都会导致这种错误。错误的原因虽然多种多样,但归根结底还是由于在执行特定于类型的操作时,变量的类型并不符合要求所致。下面来看几个例子。
var o = new 10; //抛出 typeerror
alert("name" in true); //抛出 typeerror
function.prototype.tostring.call("name"); //抛出 typeerror
最常发生类型错误的情况,就是传递给函数的参数事先未经检查,结果传入类型与预期类型不相符。
urierror 类型的错误会在使用 encodeuri()
或 decodeuri()
,而 uri 格式不正确时,就会导致 urierror 错误。这种错误也很少见,因为前面说的这两个函数的容错性非常高。
要想知道错误的类型,可以像下面这样在 try-catch
语句的 catch
语句中使用 instanceof
操作符:
try{
somefunction();
} catch (error){
if (error instanceof typeerror){
//处理类型错误
} else if (error instanceof referenceerror){
//处理引用错误
} else {
//处理其他类型的错误
}
}
在跨浏览器编程中,检查错误类型是确定处理方式的最简便途径;包含在 message
属性中的错误消息会因浏览器而异。
合理使用 try-catch
当 try-catch
语句中发生错误时,浏览器会认为错误已经被处理了,因而不会通过本章前面讨论的机制记录或报告错误。对于那些不要求用户懂技术,也不需要用户理解错误的 web 应用程序,这应该说是个理想的结果。不过,try-catch
能够让我们实现自己的错误处理机制。
使用 try-catch
最适合处理那些我们无法控制的错误。假设你在使用一个大型 javascript 库中的函数,该函数可能会有意无意地抛出一些错误。由于我们不能修改这个库的源代码,所以大可将对该函数的调用放在 try-catch
语句当中,万一有什么错误发生,也好恰当地处理它们。
在明明白白地知道自己的代码会发生错误时,再使用 try-catch
语句就不太合适了。例如,如果传递给函数的参数是字符串而非数值,就会造成函数出错,那么就应该先检查参数的类型,然后再决定如何去做。在这种情况下,不应用使用 try-catch
语句。