Node.js面试题之2017

rtwt1398 2年前
   <p>从 <strong>ECMAScript标准</strong> , <strong>Node.js语法</strong> 以及 <strong>NPM模块</strong> 角度来看,Node.js的发展让人目不暇接,那么面试题也得与时俱进。</p>    <h3>问题</h3>    <ul>     <li>什么是错误优先的回调函数?</li>     <li>如何避免回调地狱?</li>     <li>什么是Promise?</li>     <li>用什么工具保证一致的代码风格?为什么要这样?</li>     <li>什么是Stub?举例说明</li>     <li>什么是测试金字塔?举例说明</li>     <li>最喜欢哪个HTTP框架?为什么?</li>     <li>Cookies如何防范XSS攻击?</li>     <li>如何保证依赖的安全性?</li>    </ul>    <h3>答案</h3>    <p>1. 什么是错误优先的回调函数?</p>    <p>错误优先的回调函数(Error-First Callback)用于同时返回错误和数据。第一个参数返回错误,并且验证它是否出错;其他参数用于返回数据。</p>    <pre>  <code class="language-javascript">fs.readFile(filePath, function(err, data)  {   if (err)   {   // 处理错误   return console.log(err);   }   console.log(data);  });  </code></pre>    <p>2. 如何避免回调地狱?</p>    <p>以下方式可以避免回调地狱:</p>    <ul>     <li>模块化: 将回调函数转换为独立的函数</li>     <li>使用流程控制库,例如 aync</li>     <li>使用Promise</li>     <li>使用aync/await(参考 Async/Await替代Promise的6个理由 )</li>    </ul>    <p>3. 什么是Promise?</p>    <p>Promise可以帮助我们更好地处理异步操作。下面的示例中,100ms后会打印result字符串。 <strong>catch</strong> 用于错误处理。多个Promise可以链接起来。</p>    <pre>  <code class="language-javascript">new Promise((resolve, reject) =>   {   setTimeout(() =>   {   resolve('result');   }, 100)   })   .then(console.log)   .catch(console.error);  </code></pre>    <p>4. 用什么工具保证一致的代码风格?为什么要这样?</p>    <p>团队协作时,保证一致的代码风格是非常重要的,这样团队成员才可以更快地修改代码,而不需要每次去适应新的风格。这些工具可以帮助我们:</p>    <ul>     <li><a href="/misc/goto?guid=4958870940891400873" rel="nofollow,noindex">ESLint</a></li>     <li><a href="/misc/goto?guid=4959747012101840512" rel="nofollow,noindex">Standard</a></li>    </ul>    <p>5. 什么是Stub?举例说明</p>    <p>Stub用于模拟模块的行为。测试时,Stub可以为函数调用返回模拟的结果。比如说,当我们写文件时,实际上并不需要真正去写。</p>    <pre>  <code class="language-javascript">var fs = require('fs');    var writeFileStub = sinon.stub(fs, 'writeFile', function(path, data, cb)  {   return cb(null);  });    expect(writeFileStub).to.be.called;  writeFileStub.restore();  </code></pre>    <p>6. 什么是测试金字塔?举例说明</p>    <p>测试金字塔反映了需要写的 <strong>单元测试</strong> 、 <strong>集成测试</strong> 以及 <strong>端到端测试</strong> 的比例:</p>    <p style="text-align:center"><img src="https://simg.open-open.com/show/c53a2eb507dcf761f03e7ef1a5f9ce15.png"></p>    <p>测试HTTP接口时应该是这样的:</p>    <ul>     <li>很多单元测试,分别测试各个模块(依赖需要stub)</li>     <li>较少的集成测试,测试各个模块之间的交互(依赖不能stub)</li>     <li>少量端到端测试,去调用真正地接口(依赖不能stub)</li>    </ul>    <p>7. 最喜欢哪个HTTP框架?为什么?</p>    <p>这个问题标准答案。需要描述框架的优缺点,这样可以反映开发者对框架的熟悉程度。</p>    <p>8. Cookies如何防范XSS攻击?</p>    <p>XSS(Cross-Site Scripting,跨站脚本攻击)是指攻击者在返回的HTML中插入JavaScript脚本。为了减轻这些攻击,需要在HTTP头部配置 <strong>set-cookie</strong> :</p>    <ul>     <li>HttpOnly - 这个属性可以防止 <strong>cross-site scripting</strong> ,因为它会禁止Javascript脚本访问cookie。</li>     <li>secure - 这个属性告诉浏览器仅在请求为HTTPS时发送cookie。</li>    </ul>    <p>结果应该是这样的: <strong> Set-Cookie: sid= ; HttpOnly </strong> . 使用Express的话, cookie-session 默认配置好了。</p>    <p>9. 如何保证依赖的安全性?</p>    <p>编写Node.js应用时,很可能依赖成百上千的模块。例如,使用了Express的话,会直接依赖 27个模块 。因此,手动检查所有依赖是不现实的。唯一的办法是对依赖进行自动化的安全检查,有这些工具可供选择:</p>    <ul>     <li>npm outdated</li>     <li><a href="/misc/goto?guid=4959736795022340553" rel="nofollow,noindex">Trace by RisingStack</a></li>     <li><a href="/misc/goto?guid=4959741541550245313" rel="nofollow,noindex">NSP</a></li>     <li><a href="/misc/goto?guid=4959741541631263584" rel="nofollow,noindex">GreenKeeper</a></li>     <li><a href="/misc/goto?guid=4959741541721861039" rel="nofollow,noindex">Snyk</a></li>    </ul>    <h3>附加题</h3>    <p>1. 这段代码有什么问题?</p>    <pre>  <code class="language-javascript">new Promise((resolve, reject) =>   {   throw new Error('error')   })   .then(console.log)  </code></pre>    <p>then之后没有 <strong>catch</strong> 。这样的话,错误会被忽略。可以这样解决问题:</p>    <pre>  <code class="language-javascript">new Promise((resolve, reject) =>   {   throw new Error('error')   })   .then(console.log).catch(console.error)  </code></pre>    <p>调试一个大型的项目时,可以使用监控 <strong>unhandledRejection</strong> 事件来捕获所有未处理的Promise错误:</p>    <pre>  <code class="language-javascript">process.on('unhandledRejection', (err) =>  {   console.log(err)  })  </code></pre>    <h3>2. 这段代码有什么问题?</h3>    <pre>  <code class="language-javascript">function checkApiKey(apiKeyFromDb, apiKeyReceived)  {   if (apiKeyFromDb === apiKeyReceived)   {   return true   }   return false  }  </code></pre>    <p>比较密码时,不能泄露任何信息,因此比较必须在固定时间完成。否则,可以使用 timing attacks 来攻击你的应用。 <strong>为什么会这样呢</strong> ?Node.js使用V8引擎,它会从性能角度优化代码。它会逐个比较字符串的字母,一旦发现不匹配时就停止比较。当攻击者的密码更准确时,比较的时间越长。因此,攻击者可以通过比较的时间长短来判断密码的正确性。使用 cryptiles 可以解决这个问题:</p>    <pre>  <code class="language-javascript">function checkApiKey(apiKeyFromDb, apiKeyReceived)  {   return cryptiles.fixedTimeComparison(apiKeyFromDb, apiKeyReceived)  }  </code></pre>    <h3>3. 这段代码的输出是什么?</h3>    <pre>  <code class="language-javascript">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)  </code></pre>    <p>答案是2,逐行解释如下:</p>    <ol>     <li>创建新的Promise,resolve值为1。</li>     <li>x为1,加1之后返回2。</li>     <li>x为2,但是没有用到。抛出一个错误。</li>     <li>捕获错误,但是没有处理。返回1。</li>     <li>x为1,加1之后返回2。</li>     <li>x为2,打印2。</li>     <li>不会执行,因为没有错误抛出。</li>    </ol>    <p> </p>    <p> </p>    <p>来自:https://blog.fundebug.com/2017/04/10/nodejs-interview-2017/</p>    <p> </p>