随着Node.js v7.6.0版本开始支持async/await,在服务器端进行Node.js编程也终于有了最佳的异步解决方案。
只要你Node.js安装的是v7.6.x以上版本,都是支持async/await语法的。
所以只要你安装的是Node.js新版本,在Express程序里面是可以直接使用async/await方法的。
参考下面实例:
const express = require('express'); const app = express(); const fs = require('fs'); function readFileAsync(filepath) { return new Promise(function (resolve, reject) { fs.readFile(filepath, function (err, data) { if (err) { reject(err); return; } resolve(JSON.parse(data.toString())); }); }); } app.get('/', async (req, res, next) => { const data = await readFileAsync('./test.json'); res.send(data.worlds); }); // Error Handler app.use(function (err, req, res, next) { console.error('Error:', err); res.status(500).send(err.message); }); const server = app.listen(3000, () => { let port = server.address().port; console.log(`server is running on port ${port}`); });
代码可以正常执行,通过实例发现不对express进行任何改造async/await也能使用,现在我们读取一个不存在的文件看看存在的问题。
const data = await readFileAsync('./test2.json');
怎么才能捕获async函数里面的错误呢?
解决方法:
1,添加try/catch
app.get('/', async (req, res, next) => { try{ const data = await readFileAsync('./test2.json'); res.send(data.worlds); }catch(err){ next(err); } });
运行发现此时错误可以被错误中间件捕获处理了,如果在所有可能出现错误的地方的都添加try/catch,代码看起来就太不优雅精简了,更好一点的方式就是写一个方法可以自动捕获async函数里面的错误。
2,使用Promise对象
因为async函数默认返回的是一个Promise对象,所以可以用下面的asyncHandler函数捕获async函数中出现的异常错误。
//执行捕获异常错误 const asyncHandler = function (fn) { return function (...args) { Promise.resolve(fn(...args)).catch(args[2]); } } app.get('/', asyncHandler(async (req, res, next) => { const data = await readFileAsync('./test2.json'); res.send(data.worlds); }));
不过这些实现代码看起来还是怪怪的,每个用到的地方都要添加asyncHandler函数,这是作为一个有追求的程序员不能忍受的,我们要追求更极致的方式。
3,更精简的实现方式是我们可以通过修改express的底层router来实现:
const Layer = require('express/lib/router/layer'); Object.defineProperty(Layer.prototype, 'handle', { enumerable: true, get() { return this.__handle; }, set(fn) { if (fn.length === 4) { this.__handle = fn; } else { this.__handle = (req, res, next) => Promise.resolve(fn(req, res, next)).catch(next); } }, });
这样修改之后就可以全局捕获express路由async函数中出现的异常错误了,下面是完整的代码:
const express = require('express'); const app = express(); const fs = require('fs'); function readFileAsync(filepath) { return new Promise(function (resolve, reject) { fs.readFile(filepath, function (err, data) { if (err) { reject(err); return; } resolve(JSON.parse(data.toString())); }); }); } const Layer = require('express/lib/router/layer'); Object.defineProperty(Layer.prototype, 'handle', { enumerable: true, get() { return this.__handle; }, set(fn) { if (fn.length === 4) { this.__handle = fn; } else { this.__handle = (req, res, next) => Promise.resolve(fn(req, res, next)).catch(next); } }, }); app.get('/', async (req, res, next) => { const data = await readFileAsync('./test2.json'); res.send(data.worlds); }); // Error Handler app.use(function (err, req, res, next) { console.error('Error:', err); res.status(500).send(err.message); }); const server = app.listen(3000, () => { let port = server.address().port; console.log(`server is running on port ${port}`); });
so easy...^_^