当前位置: 移动技术网 > IT编程>脚本编程>NodeJs > 一些可能会用到的Node.js面试题

一些可能会用到的Node.js面试题

2019年07月19日  | 移动技术网IT编程  | 我要评论

前言

从ecmascript标准,node.js语法以及npm模块角度来看,node.js的发展让人目不暇接,那么面试题也得与时俱进。

问题

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

答案

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

错误优先的回调函数(error-first callback)用于同时返回错误和数据。第一个参数返回错误,并且验证它是否出错;其他参数用于返回数据。

fs.readfile(filepath, function(err, data)
{
if (err)
{
// 处理错误
return console.log(err);
}
console.log(data);
});

2. 如何避免回调地狱?

以下方式可以避免回调地狱:

  • 模块化: 将回调函数转换为独立的函数
  • 使用流程控制库,例如aync
  • 使用promise
  • 使用aync/await

3. 什么是promise?

promise可以帮助我们更好地处理异步操作。下面的示例中,100ms后会打印result字符串。catch用于错误处理。多个promise可以链接起来。

new promise((resolve, reject) =>
{
settimeout(() =>
{
resolve('result');
}, 100)
})
.then(console.log)
.catch(console.error);

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

团队协作时,保证一致的代码风格是非常重要的,这样团队成员才可以更快地修改代码,而不需要每次去适应新的风格。这些工具可以帮助我们:

5. 什么是stub?举例说明

stub用于模拟模块的行为。测试时,stub可以为函数调用返回模拟的结果。比如说,当我们写文件时,实际上并不需要真正去写。

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. 什么是测试金字塔?举例说明

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

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

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

7. 最喜欢哪个http框架?为什么?

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

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个模块。因此,手动检查所有依赖是不现实的。唯一的办法是对依赖进行自动化的安全检查,有这些工具可供选择:

  • npm outdated
  • trace by risingstack
  • nsp
  • greenkeeper
  • snyk

附加题

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. 不会执行,因为没有错误抛出。

英文: node.js interview questions and answers (2017 edition)

译者: fundebug

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持移动技术网。

如对本文有疑问, 点击进行留言回复!!

相关文章:

验证码:
移动技术网