Node.js面试题之2017

译者按:ECMAScript标准Node.js语法以及NPM模块角度来看,Node.js的发展让人应接不暇,那么面试题也得与时俱进。javascript

为了保证可读性,本文采用意译而非直译。html

问题

  • 什么是错误优先的回调函数?
  • 如何避免回调地狱?
  • 什么是Promise?
  • 用什么工具保证一致的代码风格?为何要这样?
  • 什么是Stub?举例说明
  • 什么是测试金字塔?举例说明
  • 最喜欢哪一个HTTP框架?为何?
  • Cookies如何防范XSS攻击?
  • 如何保证依赖的安全性?

答案

1. 什么是错误优先的回调函数?

错误优先的回调函数(Error-First Callback)用于同时返回错误和数据。第一个参数返回错误,而且验证它是否出错;其余参数用于返回数据。java

fs.readFile(filePath, function(err, data) {
    if (err)
    {
        // 处理错误
        return console.log(err);
    }
    console.log(data);
});
复制代码

2. 如何避免回调地狱?

如下方式能够避免回调地狱:node

3. 什么是Promise?

Promise能够帮助咱们更好地处理异步操做。下面的示例中,100ms后会打印result字符串。catch用于错误处理。多个Promise能够连接起来。git

new Promise((resolve, reject) =>
    {
        setTimeout(() =>
        {
            resolve('result');
        }, 100)
    })
    .then(console.log)
    .catch(console.error);
复制代码

4. 用什么工具保证一致的代码风格?为何要这样?

团队协做时,保证一致的代码风格是很是重要的,这样团队成员才能够更快地修改代码,而不须要每次去适应新的风格。这些工具能够帮助咱们:github

感兴趣的话,能够参考JavaScript Clean Coding面试

5. 什么是Stub?举例说明

Stub用于模拟模块的行为。测试时,Stub能够为函数调用返回模拟的结果。好比说,当咱们写文件时,实际上并不须要真正去写。express

var fs = require('fs');

var writeFileStub = sinon.stub(fs, 'writeFile', function(path, data, cb) {
    return cb(null);
});

expect(writeFileStub).to.be.called;
writeFileStub.restore();
复制代码

6. 什么是测试金字塔?举例说明

测试金字塔反映了须要写的单元测试集成测试以及端到端测试的比例:npm

测试HTTP接口时应该是这样的:

  • 不少单元测试,分别测试各个模块(依赖须要stub)
  • 较少的集成测试,测试各个模块之间的交互(依赖不能stub)
  • 少许端到端测试,去调用真正地接口(依赖不能stub)

7. 最喜欢哪一个HTTP框架?为何?

这个问题标准答案。须要描述框架的优缺点,这样能够反映开发者对框架的熟悉程度。json

8. Cookies如何防范XSS攻击?

XSS(Cross-Site Scripting,跨站脚本攻击)是指攻击者在返回的HTML中插入JavaScript脚本。为了减轻这些攻击,须要在HTTP头部配置set-cookie:

  • HttpOnly - 这个属性能够防止cross-site scripting,由于它会禁止Javascript脚本访问cookie。
  • secure - 这个属性告诉浏览器仅在请求为HTTPS时发送cookie。

结果应该是这样的: Set-Cookie: sid=; HttpOnly. 使用Express的话,cookie-session默认配置好了。

9. 如何保证依赖的安全性?

编写Node.js应用时,极可能依赖成百上千的模块。例如,使用了Express的话,会直接依赖27个模块。所以,手动检查全部依赖是不现实的。惟一的办法是对依赖进行自动化的安全检查,有这些工具可供选择:

附加题

1. 这段代码有什么问题?

new Promise((resolve, reject) =>
    {
        throw new Error('error')
    })
    .then(console.log)
复制代码

then以后没有catch。这样的话,错误会被忽略。能够这样解决问题:

new Promise((resolve, reject) =>
    {
        throw new Error('error')
    })
    .then(console.log).catch(console.error)
复制代码

调试一个大型的项目时,可使用监控unhandledRejection事件来捕获全部未处理的Promise错误:

process.on('unhandledRejection', (err) =>
{
    console.log(err)
})
复制代码

2. 这段代码有什么问题?

function checkApiKey(apiKeyFromDb, apiKeyReceived) {
    if (apiKeyFromDb === apiKeyReceived)
    {
        return true
    }
    return false
}
复制代码

比较密码时,不能泄露任何信息,所以比较必须在固定时间完成。不然,可使用timing attacks来攻击你的应用。为何会这样呢?Node.js使用V8引擎,它会从性能角度优化代码。它会逐个比较字符串的字母,一旦发现不匹配时就中止比较。当攻击者的密码更准确时,比较的时间越长。所以,攻击者能够经过比较的时间长短来判断密码的正确性。使用cryptiles能够解决这个问题:

function checkApiKey(apiKeyFromDb, apiKeyReceived) {
    return cryptiles.fixedTimeComparison(apiKeyFromDb, apiKeyReceived)
}
复制代码

3. 这段代码的输出是什么?

Promise.resolve(1)  
  .then((x) => x + 1)
  .then((x) => { throw new Error('My Error') })
  .catch(() => 1)
  .then((x) => x + 1)
  .then((x) => console.log(x))
  .catch(console.error)
复制代码

答案是2,逐行解释以下:

  1. 建立新的Promise,resolve值为1。
  2. x为1,加1以后返回2。
  3. x为2,可是没有用到。抛出一个错误。
  4. 捕获错误,可是没有处理。返回1。
  5. x为1,加1以后返回2。
  6. x为2,打印2。
  7. 不会执行,由于没有错误抛出。

参考连接

关于Fundebug

Fundebug专一于JavaScript、微信小程序、微信小游戏、支付宝小程序、React Native、Node.js和Java实时BUG监控。 自从2016年双十一正式上线,Fundebug累计处理了7亿+错误事件,获得了Google、360、金山软件、百姓网等众多知名用户的承认。欢迎免费试用!

版权声明

转载时请注明做者Fundebug以及本文地址:
blog.fundebug.com/2017/04/10/…

相关文章
相关标签/搜索