1、在开发Node.js的时候异步的嵌套很麻烦,有人叫回调地狱或回调黑洞,如何解决这一问题?
答:promise async await
多层嵌套:
const express = require('express');
const app = express();
const request = require('request');
app.get('/data', function(req, res) {
request('php地址1', funciton(error, response, body) {
request('php地址3', function(error, response, body) {
request('php地址3', function(error, response, body) {
...
res.json({
data: body
})
})
})
})
})
使用async和await:
const express = require('express');
const app = express();
const request = require('request');
const rp = require('request-promise');
app.get('/data', async function(req, res) {
const result1 = await rp('php地址1');
const result2 = await rp('php地址2');
const result3 = await rp('php地址3');
console.log(result1 + result2 + result3);
})
2、如何解释Node.js适合IO密集型不适合CPU密集型?
答:
- CPU密集型:计算等逻辑判断的操作,如:压缩、解压、加密和解密等。
-
I/O密集型:存取设备,网络设施的读取操作,如:文件的存取,http等网络操作,数据库操作等。
Node.js 执行是单线程的,如果执行 CPU 密集的任务就会阻塞后续代码,且单线程无法充分利用 CPU 多核资源。而异步 I/O 是多线程的,在工作线程上执行,不会阻塞执行线程,执行效率更高。
3、画出Node.js的异步时间回调机制的实现,并解释原理。
答:如下图:
原理:
(1)V8引擎解析JavaScript脚本。
(2)解析后的代码,调用Node API。
(3)libuv库负责Node API的执行。它将不同的任务分配给不同的线程,形成一个Event Loop(事件循环),以异步的方式将任务的执行结果返回给V8引擎。
(4)V8引擎再将结果返回给用户。
4、开发一个完整的Node.js程序,有UI层、service层、DAO层、MODEL层等,我们要在交给QA测试前需要编写测试用例。测试用例一般遵循测试金字塔(测试金字塔指的是编写测试用例时,底层的单元测试应该永远比上层的端到端测试用例要多,如下图),请问如下三个阶段,都用了什么样的技术进行测试用例的编写?
答:
unit 指的是一小块业务逻辑,可以使一个小组件、一个小函数、一个小类等
通常用karma+jasmine测试
service 主要是用 mocha+chai 异步测试
UI 主要是用 selenium-webdriver或者nightwatch
5、有人说Node.js是玩具,写错一处整个网站都挂掉,有什么解决办法?
答:
① 使用errorHandler处理对应的错误,如404等
② 用SafeRequest替代request
③ 容错,PM2 0秒热启功能(守护进程)
④ 电话,邮件,日志等出错的通知机制
⑤ 用process提供了一个事件uncaughtException进行抓取异常进行处理
process.on('uncaughtException', function(err){
console.log(err);
console.log(err.stack);
});
⑥ 企业级错误监测产品,如腾讯的bugly
6、写出知道的HTTP常用请求报头,并写出常见的HTTP status code含义?
答:常用的HTTP请求报头(Request Headers)
① Accept-Charset:标明浏览器可以使用的字符集(如ISO-8859-1).
② Accept-Encoding:详细列出客户端能处理的编码类型(gzip,compress是两种常见的值),一般来讲花在解码上的时间要远远小于传输的开销。
③ Accept-Language:在servlet能够以多种语言生产结果时,列出客户程序首选的语言。这个报头的值应该是标准语言代码的一种,比如en, en-us, da等。
④ Connection:标明客户是否能够处理持续性HTTP连接。持续性(Keep-Alive)是默认的选项。
⑤ Content-Length:只适用于POST请求,用来给定POST数据的大小,以字节为单位。
⑥ Cookie:这个报头向服务器返回cookie,这些cookie是之前服务器发送给浏览器的。使用request.getCookies读取这个报头。
⑦ Host:在HTTP1.1中,浏览器和其他客户端程序需要制定这个报头,它标明原始URL中给出的主机名和端口号。
⑧ Use-Agent标识生产请求的浏览器或其他客户端程序,根据这个报头,可以针对不同类型的浏览器返回不同的内容。
7、请用Koa2实现基本的服务端,并输出Hello world,且实现功能测试。
答:代码如下:
const Koa = require('koa');
const app = new Koa();
app.use(ctx => {
ctx.body = Hello world'';
}
app.listen(3000);
功能测试代码:
const app = require('app');
const request = require('supertest').agent(app.listen());
describe('Hello world', function(){
it('should say "Hello world" ', function(done){
request
.get('/')
.expect(200)
.expect('Hello world', done);
})
})
8、描述何种情况下会造成Node.js的内存泄漏,如何检测?
答:① 全局变量
a = 1;
这种比较简单的原因,全局变量直接挂在 root 对象上,不会被清除掉。
② 闭包
function outter(){
var a = 1;
return function inner(b){
console.log(a+b);
}
}
var outter2 = outter();
outter2(5);
闭包会引用到父级函数中的变量,如果闭包未释放,就会导致内存泄漏。
③ 过大的数组循环
var arr=[];
for(var i = 0; i<10000; i++){
arr.push(i);
}
9、描述一下当下新时代的前端中Node.js的作用及其实际应用?以及一般Node.js项目文件夹的划分?
答:作用及应用:
① 作为数据中间代理层,转发数据
② 为前端提供路由
③ 封装后端接口
文件夹的划分:
models:请求数据
controller:路由配置
public:静态资源
views:模板文件
libs:公共文件
config:配置文件
test:测试文件
app.js启动文件
10、浏览器缓存机制。
答:图片来自:浏览器缓存机制
浏览器第一次请求:
浏览器再次请求时: