前端常见面试题(八)@郝晨光
2019-07-11 本文已影响181人
郝晨光
ES5/ES6 的继承除了写法以外还有什么区别?
- ES5寄生组合式继承(只是列举一个方法,ES5继承还有很多实现方式)
function Parent(name) {
this.name = name;
}
Parent.prototype.say = function() {
console.log(this.name);
};
function Child(name,sex) {
Parent.call(this,name);
this.sex = sex;
}
const ChildProtoType = Object.create(Parent.prototype);
ChildProtoType.constructor = Child;
Child.prototype = ChildProtoType;
Child.constructor = Child;
Child.prototype.getSex = function() {
console.log(this.sex);
};
let child = new Child('张三','男');
console.log(child);
child.say();
child.getSex();
ES5继承
- ES6继承
class Parent {
constructor(name) {
this.name = name;
}
say() {
console.log(this.name);
}
}
class Child extends Parent {
constructor(name, sex) {
super(name);
this.sex = sex;
}
getSex() {
console.log(this.sex);
}
}
let child = new Child('李四', '女');
console.log(child);
child.say();
child.getSex();
ES6继承
ES6中子类继承父类的属性使用了super关键字,ES6语法实现是ES5的语法糖,表面上,ES6的类关键字和子类继承关键字 实现的结构和ES5继承一样,但是根本还是有差别的,ES5继承prototype属性是先实例化父类,直接继承;而 ES6是在实例化子类对象时继承父类的prototype,即实例化父类。
http状态码有哪些?分别是什么意思?
-
1**(信息类):表示接收到请求并且继续处理
- 100 Continue 继续,一般在发送post请求时,已发送了http header之后服务端将返回此信息,表示确认,之后发送具体参数信息
-
2**(响应成功):表示动作被成功接收、理解和接受
- 200 OK 正常返回信息
- 201 Created 请求成功并且服务器创建了新的资源
- 202 Accepted 服务器已接受请求,但尚未处理
-
3**(重定向类):为了完成指定的动作,必须接受进一步处理
- 301 Moved Permanently 请求的网页已永久移动到新位置。
- 302 Found 临时性重定向。
- 303 See Other 临时性重定向,且总是使用 GET 请求新的 URI。
- 304 Not Modified 自从上次请求后,请求的网页未修改过。
-
4**(客户端错误类):请求包含错误语法或不能正确执行
- 400 Bad Request 服务器无法理解请求的格式,客户端不应当尝试再次使用相同的内容发起请求。
- 401 Unauthorized 请求未授权。
- 403 Forbidden 禁止访问。
- 404 Not Found 找不到如何与 URI 相匹配的资源。
-
5**(服务端错误类):服务器不能正确执行一个正确的请求
- 500 Internal Server Error 最常见的服务器端错误。
- 503 Service Unavailable 服务器端暂时无法处理请求(可能是过载或维护)。
浏览器是如何渲染页面的?
-
简述浏览器渲染过程
- 解析HTML以构建DOM树:渲染引擎开始解析HTML文档,转换树中的html标签或js生成的标签到DOM节点,它被称为 — 内容树。
- 构建渲染树:解析CSS(包括外部CSS文件和样式元素以及js生成的样式),根据CSS选择器计算出节点的样式,创建另一个树 — 渲染树。
- 布局渲染树: 从根节点递归调用,计算每一个元素的大小、位置等,给每个节点所应该出现在屏幕上的精确坐标。
- 绘制渲染树: 遍历渲染树,每个节点将使用UI后端层来绘制。
-
CSS阻塞渲染
- CSS 是阻塞渲染的资源。需要将它尽早、尽快地下载到客户端,以便缩短首次渲染的时间。
- 这就是为什么我们将外部样式的引入放在head标签中的原因,在body渲染前先把相对完整CSSOM Tree构建好。
-
JavaScript阻塞渲染
- JavaScript 会阻止 DOM 构建和延缓网页渲染。 为了实现最佳性能,可以让您的JavaScript 异步执行,并去除关键渲染路径中任何不必要的 JavaScript。
详细链接:
typeof 和 instanceof 相同点与不同点
相同点:
typeof
和instanceof
都用来判断一个变量的数据类型。
不同点:
-
typeof
返回值是一个字符串, 用来说明变量的数据类型。 -
typeof
一般只能返回如下六个数据类型,null会返回Object,若参数为引用类型,始终返回object :number, boolean, string, function, object, undefined。
-
instanceof
返回一个布尔值。 -
instanceof
一般用来判断一个变量是否来自某个引用类型的实例,判断在其原型链中是否存在一个构造函数的prototype属性,如:console.log([] instanceof Array); // true console.log([] instanceof Object); // true
- 可以看到的是
[]
是来自于Array
是没有问题的,返回值为true
;但是[] instanceof Object
也为true
是我们不想看到的,这是因为在javascript中,Object是最顶级的数据类型,所有的引用类型最终都会指向Object - 对于这种情况,我们可以使用ES6新增的
Array.isArray([])
来进行判断,也可以通过[].constructor === Array
来判断
如何解决回调地狱?请手写代码
一、拆解function
function methodOne() {
fs.readFile(url, (err, content) => {
// do some thing
})
}
function methodTwo() {
fs.readFile(url, (err, content) => {
// do some thing
methodOne()
})
}
fs.readFile(url, (err, content) => {
if(!err) {
methodTwo()
}
})
二、事件发布/监听模式
前端常见面试题(七)@郝晨光 实现异步的几种方法中 3. 发布者订阅者模式;
三、Promise
function syncMethod() {
return new Promise((reslove, reject) => {
if(true) {
reslove('success! do some thing')
}else {
reject('error message!')
}
})
}
syncMethod().then(res => {
console.log(res);
return syncMethod();
}).then(res => {
console.log(res);
})
四、Generator
function *syncMethod() {
yield 1;
yield 2;
yield 3;
return 4;
}
let generator = syncMethod();
console.log(generator.next()); // {value: 1, done: false}
console.log(generator.next()); // {value: 2, done: false}
console.log(generator.next()); // {value: 3, done: false}
console.log(generator.next()); // {value: 4, done: true}
五、async/await
- function拆分的方式其实仅仅只是拆分代码块,时常会不利于后续维护;
- 事件发布/监听方式模糊了异步方法之间的流程关系;
- Promise虽然使得多个嵌套的异步调用能够通过链式的API进行操作,但是过多的then也增加了代码的冗余,也对阅读代码中各阶段的异步任务产生了一定干扰;
- 通过generator虽然能提供较好的语法结构,但是毕竟generator与yield的语境用在这里多少还有些不太贴切。
- 所以就有了async/await语法糖
// 模拟获取数据
function getData() {
return 'response';
}
// async函数
async function syncMethod() {
let res = await getData(); // 等待异步结束,将结果保存在变量中
console.log(res); // 一般异步的话,await后边会跟随一个Promise对象,调用.then方法来获取值
}
syncMethod()