JavaScript 面试
叶江怀 javascript
# JS相关
# 1. ES5和ES6继承方式区别
- ES5定义类以函数形式, 以prototype来实现继承
- ES6以class形式定义类, 以extend形式继承
# 2. Generator了解
ES6 提供的一种异步编程解决方案, Generator 函数是一个状态机,封装了多个内部状态。
function* helloWorldGenerator() {
yield 'hello';
yield 'world';
return 'ending';
}
var hw = helloWorldGenerator();
1
2
3
4
5
6
7
2
3
4
5
6
7
调用后返回指向内部状态的指针, 调用next()才会移向下一个状态, 参数:
hw.next()
// { value: 'hello', done: false }
hw.next()
// { value: 'world', done: false }
hw.next()
// { value: 'ending', done: true }
hw.next()
// { value: undefined, done: true }
1
2
3
4
5
6
7
8
9
10
11
2
3
4
5
6
7
8
9
10
11
# 3. 手写Promise实现
var myPromise = new Promise((resolve, reject) => {
// 需要执行的代码
...
if (/* 异步执行成功 */) {
resolve(value)
} else if (/* 异步执行失败 */) {
reject(error)
}
})
myPromise.then((value) => {
// 成功后调用, 使用value值
}, (error) => {
// 失败后调用, 获取错误信息error
})
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
2
3
4
5
6
7
8
9
10
11
12
13
14
15
# 4. Promise优缺点
- 优点: 解决回调地狱, 对异步任务写法更标准化与简洁化
- 缺点: 首先,无法取消Promise,一旦新建它就会立即执行,无法中途取消; 其次,如果不设置回调函数,Promise内部抛出的错误,不会反应到外部; 第三,当处于pending状态时,无法得知目前进展到哪一个阶段(刚刚开始还是即将完成). 极简版promise封装:
function promise () {
this.msg = '' // 存放value和error
this.status = 'pending'
var that = this
var process = arguments[0]
process (function () {
that.status = 'fulfilled'
that.msg = arguments[0]
}, function () {
that.status = 'rejected'
that.msg = arguments[0]
})
return this
}
promise.prototype.then = function () {
if (this.status === 'fulfilled') {
arguments[0](this.msg)
} else if (this.status === 'rejected' && arguments[1]) {
arguments[1](this.msg)
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
# 5. 观察者模式
又称发布-订阅模式, 举例子说明. 实现: 发布者管理订阅者队列, 并有新消息推送功能. 订阅者仅关注更新就行
# 6. 手写实现bind
Function.prototype.bind = function () {
// 保存原函数
var self = this
// 取出第一个参数作为上下文, 相当于[].shift.call(arguments)
var context = Array.prototype.shift.call(arguments)
// 取剩余的参数作为arg; 因为arguments是伪数组, 所以要转化为数组才能使用数组方法
var arg = Array.prototype.slice.call(arguments)
// 返回一个新函数
return function () {
// 绑定上下文并传参
self.apply(context, Array.prototype.concat.call(arg, Array.prototype.slice.call(arguments)))
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
2
3
4
5
6
7
8
9
10
11
12
13
# 7. 手写实现4种继承
function Father () {}
function Child () {}
// 1. 原型继承
Child.prototype = new Father()
// 2. 构造继承
function Child (name) {
Father.call(this, name)
}
// 3. 组合继承
function Child (name) {
Father.call(this, name)
}
Child.prototype = new Father()
// 4. 寄生继承
function cloneObj (o) {
var clone = object.create(o)
clone.sayName = ...
return clone
}
// 5. 寄生组合继承
// 6. ES6 class extend继承
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
# 9. http状态码
- 1**: 服务器收到请求, 需请求者进一步操作
- 2**: 请求成功
- 3**: 重定向, 资源被转移到其他URL了
- 4**: 客户端错误, 请求语法错误或没有找到相应资源
- 5**: 服务端错误, server error
- 304: Not Modified. 指定日期后未修改, 不返回资源
# 10. Object.create实现(原型式继承,特点:实例的__proto__指向构造函数本身)
# 11. async和await:(ES8)
- Generator函数的语法糖,将*改成async,将yield换成await。
- 是对Generator函数的改进, 返回promise。
- 异步写法同步化,遇到await先返回,执行完异步再执行接下来的.
- 内置执行器, 无需next()
# 12. 算法和数据结构:
- 算法: 解决具体问题所需要的解决方法。执行效率最快的最优算法。时间复杂度。输入,输出,有穷性,确定性,可行性。冒泡排序,二叉树遍历,最长回文,二分查找,指针,链表等,堆栈,队列等。力扣,codewar,算法导论。
- 数据结构: 逻辑结构:集合、线性、树形、图形结构 物理结构:顺序、链式存储结构
# 13. 封装JSONP
function jsonp ({url, param, callback}) {
return new Promise((resolve, reject) => {
var script = document.createElement('script')
window.callback = function (data) {
resolve(data)
document.body.removeChild('script')
}
var param = {...param, callback}
var arr = []
for (let key in param) {
arr.push(`${key}=${param[key]}`)
}
script.src = `${url}?${arr.join('&')}`
document.body.appendChild(script)
})
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
# 14. 手动实现map(forEach以及filter也类似)
// for循环实现
Array.prototype.myMap = function () {
var arr = this
var [fn, thisValue] = Array.prototype.slice.call(arguments)
var result = []
for (var i = 0; i < arr.length; i++) {
result.push(fn.call(thisValue, arr[i], i, arr))
}
return result
}
var arr0 = [1, 2, 3]
console.log(arr0.myMap(v => v + 1))
// forEach实现(reduce类似)
Array.prototype.myMap = function (fn, thisValue) {
var result = []
this.forEach((v, i, arr) => {
result.push(fn.call(thisValue, v, i, arr))
})
return result
}
var arr0 = [1, 2, 3]
console.log(arr0.myMap(v => v + 1))
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
# 15. js实现checkbox全选以及反选
<body>
<button id="other">反选</button>
<input type="checkbox" id="all" />全选
<input type="checkbox" class="check" />1
<input type="checkbox" class="check" />2
<input type="checkbox" class="check" />3
<script>
var checkbox = document.getElementsByClassName('check')
var checkAll = document.getElementById('all')
var checkOther = document.getElementById('other')
checkAll.onclick = function() {
var flag = true
for (var i = 0; i < checkbox.length; i++) {
if (!checkbox[i].checked) flag = false
}
if (flag) {
for (var i = 0; i < checkbox.length; i++) {
checkbox[i].checked = false
}
} else {
for (var i = 0; i < checkbox.length; i++) {
checkbox[i].checked = true
}
}
}
checkOther.onclick = function() {
for (var i = 0; i < checkbox.length; i++) {
checkbox[i].checked = !checkbox[i].checked
}
}
</script>
</body>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
# 16. 对原型链的理解?prototype上都有哪些属性
- 在js里,继承机制是原型继承。继承的起点是 对象的原型(Object prototype)。
- 一切皆为对象,只要是对象,就会有 proto 属性,该属性存储了指向其构造的指针。
- Object prototype也是对象,其 proto 指向null。
- 对象分为两种:函数对象和普通对象,只有函数对象拥有『原型』对象(prototype)。
- prototype的本质是普通对象。
- Function prototype比较特殊,是没有prototype的函数对象。
- new操作得到的对象是普通对象。
- 当调取一个对象的属性时,会先在本身查找,若无,就根据 proto 找到构造原型,若无,继续往上找。最后会到达顶层Object prototype,它的 proto 指向null,均无结果则返回undefined,结束。
- 由 proto 串起的路径就是『原型链』。
- 通过prototype可以给所有子类共享属性
- 函数(Function)才有prototype属性,对象(除Object)拥有__proto__
- 实例的__proto__等于构造函数的prototype
- 实例的constructor指向构造函数
# 17. 为什么使用继承
通常在一般的项目里不需要,因为应用简单,但你要用纯js做一些复杂的工具或框架系统就要用到了,比如webgis、或者js框架如jquery、ext什么的,不然一个几千行代码的框架不用继承得写几万行,甚至还无法维护
# 18. setTimeout时间延迟为何不准
单线程, 先执行同步主线程, 再执行异步任务队列
# 19. 事件循环述,宏任务和微任务有什么区别?
- 先主线程后异步任务队列
- 先微任务再宏任务
# 20. let const var作用域
块级作用域, 暂时性死区
# 21. 节流和防抖
- 函数节流是指一定时间内js方法只跑一次。比如人的眨眼睛,就是一定时间内眨一次。这是函数节流最形象的解释。
// 函数节流 滚动条滚动
var canRun = true;
document.getElementById("throttle").onscroll = function(){
if(!canRun){
// 判断是否已空闲,如果在执行中,则直接return
return;
}
canRun = false;
setTimeout(function(){
console.log("函数节流");
canRun = true;
}, 300);
};
1
2
3
4
5
6
7
8
9
10
11
12
13
14
2
3
4
5
6
7
8
9
10
11
12
13
14
- 函数防抖是指频繁触发的情况下,只有足够的空闲时间,才执行代码一次。比如生活中的坐公交,就是一定时间内,如果有人陆续刷卡上车,司机就不会开车。只有别人没刷卡了,司机才开车。
// 函数防抖
var timer = false;
document.getElementById("debounce").onscroll = function(){
clearTimeout(timer); // 清除未执行的代码,重置回初始化状态
timer = setTimeout(function(){
console.log("函数防抖");
}, 300);
};
1
2
3
4
5
6
7
8
9
2
3
4
5
6
7
8
9
# 22. 实现一个sleep函数
// 这种实现方式是利用一个伪死循环阻塞主线程。因为JS是单线程的。所以通过这种方式可以实现真正意义上的sleep()。
function sleep(delay) {
var start = (new Date()).getTime();
while ((new Date()).getTime() - start < delay) {
continue;
}
}
function test() {
console.log('111');
sleep(2000);
console.log('222');
}
test()
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
2
3
4
5
6
7
8
9
10
11
12
13
14
15
# 23. 闭包
- 概念: 内层函数能够访问外层函数作用域的变量
- 缺点: 引起内存泄漏(释放内存)
- 作用:
- 保护this指向
- 使用闭包修正打印值
- 实现柯里化
- 实现node commonJs 模块化, 实现私有变量
- 保持变量与函数活性, 可延迟回收和执行
# 24. Immutable.js
Facebook出品, 倡导数据的不可变性, 用的最多就是List和Map.
# 25. js实现instanceof
// 检测l的原型链(__proto__)上是否有r.prototype,若有返回true,否则false
function myInstanceof (l, r) {
var R = r.prototype
while (l.__proto__) {
if (l.__proto__ === R) return true
}
return false
}
1
2
3
4
5
6
7
8
2
3
4
5
6
7
8
# 27. ES6的模块引入和CommonJs区别
# 28. 严格模式
// 严格模式下, 隐式绑定丢失后this不会指向window, 而是指向undefined
'use strict'
var a = 2
var obj = {
a: 1,
b: function() {
// console.log(this.a)
console.log(this)
}
}
var c = obj.b
c() // undefined
1
2
3
4
5
6
7
8
9
10
11
12
2
3
4
5
6
7
8
9
10
11
12
# 30. typescript缺点
- 并不是严格意义的js的超集, 与js不完全兼容, 会报错
- 更多的限制, 是一种桎梏
- 有些js第三方库没有dts, 有问题
# 31. 构造函数实现原理(new操作符做了什么)
- 构造函数中没有显示的创建Object对象, 实际上后台自动创建了一个空对象
- 直接给this对象赋值属性和方法, this即指向创建的对象
- 没有return返回值, 后台自动返回了该对象
- 该对象继承构造函数的原型
// 模拟构造函数实现
var Book = function(name) {
this.name = name;
};
//正常用法
var java = new Book(‘Master Java’);
//使用代码模拟,在非IE浏览器中测试,IE浏览器不支持
var python = {};
python.__proto__ = Book.prototype;
Book.call(python, 'Master Python');
1
2
3
4
5
6
7
8
9
10
11
12
2
3
4
5
6
7
8
9
10
11
12
# 32. for in 和 for of区别
for in
遍历数组会遍历到数组原型上的属性和方法, 更适合遍历对象forEach
不支持break, continue, return
等- 使用
for of
可以成功遍历数组的值, 而不是索引, 不会遍历原型 - for in 可以遍历到myObject的原型方法method,如果不想遍历原型方法和属性的话,可以在循环内部判断一下,hasOwnPropery方法可以判断某属性是否是该对象的实例属性
# 33. JS实现并发控制:
使用消息队列以及setInterval
或promise
进行入队和出队
# 34. ajax和axios、fetch的区别
# 35. promise.finally实现
Promise.prototype.finally = function (callback) {
let P = this.constructor;
return this.then(
value => P.resolve(callback()).then(() => value),
reason => P.resolve(callback()).then(() => { throw reason })
);
};
1
2
3
4
5
6
7
2
3
4
5
6
7
# 36. 实现symbol
(function() {
var root = this;
var SymbolPolyfill = function Symbol(description) {
// Symbol函数前不能使用new命令
if (this instanceof SymbolPolyfill) throw new TypeError('Symbol is not a constructor')
// 参数是对象,则调用toString
var descString = description === undefined ? undefined : String(description)
var symbol = Object.create({
// 显示转为字符串
toString: function() {
return 'Symbol(' + this.description ')'
},
// 不能参与运算
valueOf: function() {
throw new Error('Cannot convert a Symbol value')
}
})
// 作为对象键名时,生成一个独一无二名称
// ES2019新增
symbol.prototype.description = descString
// 返回一个新对象 由于指针不同 所以两两不等
return symbol;
}
root.SymbolPolyfill = SymbolPolyfill;
})();
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
# 37. ES6装饰器
- 提案经过大幅修改,没有定案,可能会变
- 是一种函数,用于改变类或方法的功能,起到注释作用
- 函数:
@decorator
class A {}
// 等同于
class A {}
A = decorator(A) || A;
1
2
3
4
5
6
7
2
3
4
5
6
7
- 方法:
function readonly(target, name, descriptor){
// descriptor对象原来的值如下
// {
// value: specifiedFunction,
// enumerable: false,
// configurable: true,
// writable: true
// };
descriptor.writable = false;
return descriptor;
}
readonly(Person.prototype, 'name', descriptor);
// 类似于
Object.defineProperty(Person.prototype, 'name', descriptor);
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
2
3
4
5
6
7
8
9
10
11
12
13
14
15
- 先从外到内进入,经过 reverse 倒序后,然后由内向外执行。
# 38. 实现fetch abort
const controller = new AbortController()
setTimeout(() => controller.abort(), 1000);
fetch(url, {
signal: controller.signal
}).then(res => {
...
}, err => {
console.log(err) // AbortError
})
1
2
3
4
5
6
7
8
9
10
2
3
4
5
6
7
8
9
10
# 39. 使用xhr实现fetch
- fetch是ajax替代品,基于promise。
- 类似于jquey ajax,但不是ajax封装,而是原生js,,没有使用XHR对象
- 比ajax方便,但仍然不完善,建议使用axios库
- fetch有而axios没有的问题:
- 兼容性问题
- 默认不带cookie,需要手动设置credentials
- 跨域问题,需要手动设置mode
- 返回400、500不识别为reject
- 无法abort和timeout
- 用XHR实现fetch:
// 先实现ajax
function ajax(method,url,data,suc,fail) {
var xhr = new XMLHttpRequest();
xhr.open(method, url, true);
xhr.onreadystatechange = function () {
if(xhr.readyState == 4){
if(xhr.status == 200){
suc(xhr.responseText)
} else {
console.log(err);
fail(xhr.responseText);
}
}
};
xhr.send(data);
}
// 再实现promise
function promise () {
this.msg = '' // 存放value和error
this.status = 'pending'
var that = this
var process = arguments[0]
process (function () {
that.status = 'fulfilled'
that.msg = arguments[0]
}, function () {
that.status = 'rejected'
that.msg = arguments[0]
})
return this
}
promise.prototype.then = function () {
if (this.status === 'fulfilled') {
arguments[0](this.msg)
} else if (this.status === 'rejected' && arguments[1]) {
arguments[1](this.msg)
}
}
// 最后实现fetch
function fetch(method, url, data) {
return new promise(function (resolve,reject) {
ajax(method, url, data, function (res) {
resolve(res);
},function (err) {
reject(err);
})
})
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
# 40. JS中数值存储
- javascript 中所有的数值类型都是双精度存储的,使用 64bit,64bit 等于 8byte
- 中文占2个字节(byte) 中文里标点也占2个字节(byte) 英文里的子母不分大小写,一个子母占1个字节(字节) 英文里的标点占1个字节。 1字节(byte)=8位(bits)
- 当任何数字在进行位运算时 js 内部会将其转换成32位有符号整型
# 41. this指向:
- 谁调用了方法,该方法的this就指向谁;
- 优先级:箭头函数 > new绑定 > 显示绑定 > 隐式绑定
- 1. ES5和ES6继承方式区别
- 2. Generator了解
- 3. 手写Promise实现
- 4. Promise优缺点
- 5. 观察者模式
- 6. 手写实现bind
- 7. 手写实现4种继承
- 9. http状态码
- 10. Object.create实现(原型式继承,特点:实例的_proto_指向构造函数本身)
- 11. async和await:(ES8)
- 12. 算法和数据结构:
- 13. 封装JSONP
- 14. 手动实现map(forEach以及filter也类似)
- 15. js实现checkbox全选以及反选
- 16. 对原型链的理解?prototype上都有哪些属性
- 17. 为什么使用继承
- 18. setTimeout时间延迟为何不准
- 19. 事件循环述,宏任务和微任务有什么区别?
- 20. let const var作用域
- 21. 节流和防抖
- 22. 实现一个sleep函数
- 23. 闭包
- 24. Immutable.js
- 25. js实现instanceof
- 27. ES6的模块引入和CommonJs区别
- 28. 严格模式
- 30. typescript缺点
- 31. 构造函数实现原理(new操作符做了什么)
- 32. for in 和 for of区别
- 33. JS实现并发控制:
- 34. ajax和axios、fetch的区别
- 35. promise.finally实现
- 36. 实现symbol
- 37. ES6装饰器
- 38. 实现fetch abort
- 39. 使用xhr实现fetch
- 40. JS中数值存储
- 41. this指向: