从JavaScript到Python之异常

不少前端工程师看到这个标题可能会产生质问:

我js用得好好的,能后端能APP,为什么还要学习Python?

至少有下面两个理由:

  1. 学习曲线。ES6之后的JavaScript(TypeScript)的在语法上和Python有很多异曲同工的地方,所以学习曲线非常平滑,上手难度很低。
  2. 应用场景。JavaScript虽然是web开发的王者,但是其在某些方面还是存在一些问题的。比如Node.js虽然可以写后端,但大多用于CGI层,做接口整合和模板渲染,而Python、Java这类语言已被广泛应用于后端服务编写,同时Python在做机器学习等方面也很有优势。

这篇文章我们通过对比两种语言之间的“异常”来学习Python。

为什么需要对异常进行处理

异常是编写代码时必须考虑的问题,但却并不是一个热门话题,少有文章提到。
尤其在web前端这一块,浏览器的强大容错能力帮工程师解决(隐藏)了大量的异常。
但如果忽视异常轻则影响功能运行,重则导致系统崩溃,造成经济损失。

异常处理

捕获

JavaScript 异常捕获关键字与 Python 一致,都是使用try

1
2
3
4
5
6
7
// JavaScript
try {
...
}
# Python
try:
...

但是两者都具有局限性:只能用来捕获同步执行代码的异常。

对于异步代码的异常捕获的情况 JavaScript 处理起来相对复杂,按运行环境就可以分为 Node.js 和 浏览器,按捕获范围又可以分为全部捕获和局部捕获。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
// 浏览器全局异常捕获
window.onError = e => {}
window.addEventListener('', e => {}) 捕获请求错误
// 浏览器Ajax异常捕获
var xhr = new XMLHttpRequest();
xhr.onload = function(e) {
if(this.status > 400){
...
}
};
// Node.js 全局异常捕获
process.on('uncaughtException', e => {})
// Node.js 回调方式异常捕获
fs.readFile('/etc/passwd', (err, data) => {
if (err) {
...
}
});

Python 的情况要简单很多,即使使用异步线程或进程,也可以在函数内部进行捕获,然后其它方式上报。

1
2
3
4
5
6
def _parse_speed(key_name, speed_list):
try:
...
t = threading.Thread(target=_parse_speed, args=(v, speed_list))
t.start()

由于异常捕获情况比较复杂,如无特殊说明,后文讨论仅指使用try关键字的同步异常捕获。

处理

再看看JavaScript几种处理异常方式。

  • catch。catch 语句块用来处理异常,处理方式也相当粗放,try 语句块内的所有异常都会被捕获。
  • finally。finally 语句块表示无论是否发生异常,语句块代码一定会被执行。

不少读者可能会以为在catch语句块后执行的代码和finally语句块作用一样,但其实是有区别的!

下面举个例子

1
2
3
4
5
6
7
8
9
10
11
12
fn = () => {
try {
throw Error()
} catch(e) {
throw Error()
return 1
} finally {
return 2
}
return 3
}
fn() // 是2不是3

也就是说 finally 语句块中的代码,不论 try 还是 catch 执行出错都会执行。

Python的异常处理:

  • except。可以指定需要处理的多个异常类型,可以有多个异常处理逻辑,写法非常灵活。
  • else。else 语句块可以当 try 语句块未发生异常时执行。
  • finally。特性同 JavaScript。

下面列举使用 except 的几个例子:

1
2
3
4
5
6
7
8
9
10
11
12
13
# 捕获异常实例
except Exception as e:
# 批量异常捕获
except (IOError, TypeError):
# 异常分类捕获:
except OSError as err:
...
except ValueError:
...
except:
...

抛出

JavaScript 中可以通过关键字 throw 来抛出一个 Error 对象。
Python中与之对应的是 raise ,用来抛出一个 Exception 实例。

虽然关键字不一样但是写法基本相似:

1
2
3
4
// JavaScript
throw Error(...)
# Python
raise Exception(...)

异常类型

对于 JavaScript 而言,谈异常类的意义不大,处理不同的异常需要手动判断,所以异常类型也比较简单。

Python的异常类型则丰富得多

总结

  • 捕获方面,JavaScript 的场景要比 Python 复杂,迁移学习 Python 很简单。
  • 处理方面,两者都支持 finally 关键字。不同的是JavaScript 的操作被简化,只能通过一个 catch 语句块来处理所有类型异常。而 Python 的做法和大多数高级语言一致,能多次、针对不同异常进行分别处理,else 关键字算是一个特色功能。
  • 抛出方面,两者只是关键字不同。

参考:


一部由众多技术专家推荐, 帮你成为具有全面能力和全局视野工程师的进阶利器—— 《了不起的JavaScript工程师》出版了! 点击下方链接即刻踏上进阶之路!


亚里士朱德 wechat
更多WEB技术分享请订阅微信公众号“WEB学习社”