大前端面试题库
碎片时间刷刷题
问答题
代码题
FlexBox 可视化学习
博客
关于
作者
福利
大前端面试题库
碎片时间刷刷题
问答题
代码题
博客
关于
作者
福利
关注本站公众号,
获取永久访问授权码
扫码关注,回复『刷题』即可
~技术问答题~
全部
js
经典高频
编程题
大厂
html
布局
css
浏览器
性能
前端框架
React
Vue
安全
数据结构和算法
网络
TCP
http
node
设计模式
工程化
业务实现
webpack
babel
持续集成
多端交互
Hybrid
git 相关
其他(标签待定区)
#
题名
1
ES6 代码转成 ES5 代码的实现思路是什么
简单
2
前端项目如何找出性能瓶颈
简单
3
如果让你去实现一个 CSRF 攻击你会怎么做?
简单
4
HTTP 中的 keep-alive 有了解吗?它和多路复用的区别
简单
5
介绍下 HTTPS 中间人攻击
简单
6
为什么 HTTP1.1 不能实现多路复用(腾讯)
简单
7
接口如何防刷
简单
8
谈谈你对 TCP 三次握手和四次挥手的理解
简单
9
简单讲解一下http2的多路复用
简单
10
说下 URL 和 URI 的区别?
简单
11
谈谈vuex原理?
简单
12
【简单概括】从输入浏览器网址到页面渲染经历了什么?
简单
13
如何与服务器保持长连接?
简单
14
defineProterty的属性值有哪些
简单
15
请说下VUE diff算法
简单
16
VUE watch和computed的区别
简单
17
手写用 ES6proxy 如何实现 arr[-1] 的访问
简单
18
node 中间层怎样做的请求合并转发?
简单
19
实现 add(1)(2)(3)
简单
20
说说TCP 滑动窗口?
简单
21
简单实现async/await中的async函数
简单
22
平时怎么学习(淘系)
简单
23
PWA原理,讲下核心点(淘系)
简单
24
简述TCP连接的过程(淘系)
简单
25
怎么用栈模拟队列,队列模拟栈,有思路吗(淘系)
简单
26
快排原理以及时间复杂度,为什么(淘系前端)
简单
27
mvvm与mvc的区别(淘系前端)
简单
28
你认为flutter与react native区别于在哪(淘系前端)
简单
29
addEventListener第三个参数是什么,还可以是别的吗(淘系前端)
简单
30
饿了么 - 说下 koa 的中间件机制
简单
31
说下单点登录-新东方
简单
32
pureComponent和FunctionComponent区别(有赞)
简单
33
项目中如何处理安全问题(阿里)
简单
34
居中为什么要使用transform(为什么不使用marginLeft/Top)(阿里)
简单
35
如何解决移动端适配1px的问题(阿里)
简单
36
Promise.all 并发限制 同时只能有5个请求(瓜子)
简单
37
src和 href 的区别(京东)
简单
38
react-router里的<Link>标签和<a>标签有什么区别(滴滴)
简单
39
react-router怎么实现路由切换(滴滴)
简单
40
React组件事件代理的原理(网易)
简单
41
请介绍Promise,异常捕获(网易)
简单
42
介绍下粘性布局(sticky)(网易)
简单
43
http1.1时如何复用tcp连接(网易)
简单
44
观察者和订阅-发布的区别,各自用在哪里(网易)
简单
45
文件上传如何做断点续传(网易)
简单
46
使用过的koa2中间件(阿里)
简单
47
master挂了的话pm2怎么处理(阿里)
简单
48
介绍SSL和TLS(寺库)
简单
49
说说网络的五层模型(寺库)
简单
50
RN的原理,为什么可以同时在安卓和IOS端运行(寺库)
简单
51
promise如何实现then处理(宝宝树)
简单
52
介绍MVP怎么组织(宝宝树)
简单
53
node文件查找优先级(宝宝树)
简单
54
npm2和npm3 有什么区别(宝宝树)
简单
55
什么时候使用“git rebase”代替“git merge”?
简单
56
告诉我 Git 中 HEAD、工作树和索引之间的区别?
简单
57
解释 Forking 工作流程的优点
简单
58
“git pull”和“git fetch”之间有什么区别?
简单
59
“拉取请求(pull request)”和“分支(branch)”之间有什么区别?
简单
60
什么是 Git 复刻(fork)?复刻(fork)、分支(branch)和克隆(clone)之间有什么区别?
简单
61
使用过git cherry-pick,有什么作用?
简单
62
git跟其他版本控制器有啥区别?
简单
63
我们在本地工程常会修改一些配置文件,这些文件不需要被提交,而我们又不想每次执行git status时都让这些文件显示出来,我们该如何操作?
简单
64
如何把本地仓库的内容推向一个空的远程仓库?
简单
65
提交时发生冲突,你能解释冲突是如何产生的吗?你是如何解决的?
简单
66
列举工作中常用的几个git命令?
简单
67
对象数组如何去重?(烈熊网络)
简单
68
移动端如何设计一个比较友好的Header组件?(携程)
简单
69
说出space-between和space-around的区别?(携程)
简单
70
React Hooks当中的useEffect是如何区分生命周期钩子的(喜马拉雅)
简单
71
React实现的移动应用中,如果出现卡顿,有哪些可以考虑的优化方案(喜马拉雅)
简单
72
如何从10000个数中找到最大的10个数(喜马拉雅)
简单
73
React实现一个防抖的模糊查询输入框
简单
74
手动封装一个请求函数,可以设置最大请求次数,请求成功则不再请求,请求失败则继续请求直到超过最大次数(流利说)
简单
75
GET和POST的区别(流利说)
简单
76
CSS3中transition和animation的属性分别有哪些(哔哩哔哩)
简单
77
forEach,map和filter的区别(哔哩哔哩)
简单
78
跨标签页的通讯方式有哪些(哔哩哔哩)
简单
79
实现一个函数判断数据类型(哔哩哔哩)
简单
80
delete 数组的 item,数组的 length 是否会 -1
简单
81
给出 ['1', '3', '10'].map(parseInt) 执行结果
简单
82
http 劫持
简单
83
HTTP劫持、DNS劫持与XSS
简单
84
xss csrf 的一切
简单
85
请说说 CSRF 攻防
简单
86
HTTP劫持、DNS劫持与XSS
简单
87
简单说下xss攻击和防御
简单
88
react 有几种事件绑定方式以及区别是什么
简单
89
redux异步中间件之间的优劣?
简单
90
redux中如何进行异步操作?
简单
91
redux与mobx的区别?
简单
92
react-redux是如何工作的?
简单
93
redux的工作流程?
简单
94
你对 Time Slice的理解?
简单
95
mixin、hoc、render props、react-hooks的优劣如何?
简单
96
React如何进行组件/逻辑复用?
简单
97
React有哪些性能优化的手段?
简单
98
React组件通信如何实现?
简单
99
setState到底是异步还是同步?
简单
100
React的请求应该放在哪个生命周期中?
简单
点击加载更多...
关于作者
zz_jesse 专注前端
掘金
我的开源项目
公众号@前端技术江湖
一个可以帮开发者成长的公众号
前端面试题库更新通知
前端学习资料、干货文章
技术交流群
交流中成长
大厂内推机会
You need to enable JavaScript to run this app.
{"initialData":{"/":{"init":true,"res":{"page":{"tdk":{"title":"大前端面试题库-面试刷题","keyword":"前端面试题,前端面试刷题","description":"这里是大前端面试题库，搜集了众多经典高频前端面试题和大厂前端面试题以及相关代码题，再也不用花力气去找前端面试题了。"}},"fetchData":{"tagList":[{"id":125,"className":"js"},{"id":153,"className":"经典高频"},{"id":144,"className":"编程题"},{"id":146,"className":"大厂"},{"id":124,"className":"html"},{"id":128,"className":"布局"},{"id":126,"className":"css"},{"id":137,"className":"浏览器"},{"id":148,"className":"性能"},{"id":139,"className":"前端框架"},{"id":140,"className":"React"},{"id":141,"className":"Vue"},{"id":147,"className":"安全"},{"id":134,"className":"数据结构和算法"},{"id":142,"className":"网络"},{"id":159,"className":"TCP"},{"id":127,"className":"http"},{"id":130,"className":"node"},{"id":143,"className":"设计模式"},{"id":131,"className":"工程化"},{"id":160,"className":"业务实现"},{"id":129,"className":"webpack"},{"id":138,"className":"babel"},{"id":132,"className":"持续集成"},{"id":154,"className":"多端交互"},{"id":152,"className":"Hybrid"},{"id":157,"className":"git 相关"},{"id":155,"className":"其他（标签待定区）"}],"iqList":[{"id":833,"newTitle":"ES6 代码转成 ES5 代码的实现思路是什么","subDes":"","level":3,"viewCount":0,"newContent":"### 参考1\n题目说的是 ES6，所以不考虑 .jsx、.ts 这类 js 拓展语言。\nES6 转 ES5 目前行业标配是用 Babel，转换的大致流程如下：\n\n解析：解析代码字符串，生成 AST；\n转换：按一定的规则转换、修改 AST；\n生成：将修改后的 AST 转换成普通代码。\n如果不用工具，纯人工的话，就是使用或自己写各种 polyfill 了。\n\n\n### 参考2\n\n说到 ES6 代码转成 ES5 代码，我们肯定会想到 Babel。所以，我们可以参考 Babel 的实现方式。\n\n那么 Babel 是如何把 ES6 转成 ES5 呢，其大致分为三步：\n\n将代码字符串解析成抽象语法树，即所谓的 AST\n对 AST 进行处理，在这个阶段可以对 ES6 代码进行相应转换，即转成 ES5 代码\n根据处理后的 AST 再生成代码字符串\n基于此，其实我们自己就可以实现一个简单的“编译器”，用于把 ES6 代码转成 ES5。\n\n比如，可以使用 @babel/parser 的 parse 方法，将代码字符串解析成 AST；使用 @babel/core 的 transformFromAstSync 方法，对 AST 进行处理，将其转成 ES5 并生成相应的代码字符串；过程中，可能还需要使用 @babel/traverse 来获取依赖文件等。\n\n如果有误，希望可以帮忙指出来，多谢。\n\n---\n原文在这里   [点这里](https://github.com/Advanced-Frontend/Daily-Interview-Question/issues/112)","seeCount":0,"answerUrl":""},{"id":832,"newTitle":"前端项目如何找出性能瓶颈","subDes":"","level":2,"viewCount":0,"newContent":"### 参考1\n\n\n性能监控体系, chrome perfermance devtools, 工作经验\n\n### 参考2\n\n\n分享一下我近期的经验，之前项目也碰到过用起来很卡的情况，就是用element ui的tab切换组件时，点击tab切换非常卡，非常耗时，在排除了网络请求和js代码执行时间过长等原因后，跑了一次perfermance，结果发现大部分时间都花费在了 DOM GC上了，分析了下原因可能时dom结构太多导致每次tab切换渲染太耗时了。由于我每个tab里面的html结构都一样，都是一个table，只是每次tab切换时请求的数据不一样，我就把table抽离出来了，放到tab组件外面，然后tab里面就空了，就没有那么多dom了，tab切换就不卡了，很流畅。（ps：tab有20-30个切换选项，本人语文水平不行，描述的不清楚，望轻喷。）\n\n### 参考3\n\n我觉得应该首先理一理会造成性能损耗的一些场景:\n\n1. 比如大列表的渲染,大量dom的渲染\n\n2.大量图片的加载,过多资源的请求.\n\n3.代码中有没有耗时的计算操作,或则大量循环.递归\n\n编写的组件过于庞大 层级过深,依赖模块过多等.\n我觉得首先就是查看请求的资源体积是否过大,如果过大考虑压缩,减少不必要的资源的请求,不必要的js代码的代码加载,用字体图标代替图片,异步加载等等.\n但是我觉得基本的优化策略(减少请求数,压缩请求资源的体积)都已经做过了,感觉性能还是没有提升,可能应该关注与代码层面的优化吧,比如过大的第三方库能不能换成轻量级的,代码中有没有很耗时的操作循环和递归,过多的分支条件语句,能不能改写以提高执行效率,简化复杂的组件逻辑,减少不必要的依赖,是否有杀鸡用了牛刀的操作等.暂时想到的就这些.\n\n---\n更多解析请看原文 [原文](https://github.com/Advanced-Frontend/Daily-Interview-Question/issues/300)\n\n\n","seeCount":0,"answerUrl":""},{"id":831,"newTitle":"如果让你去实现一个 CSRF 攻击你会怎么做？","subDes":"","level":2,"viewCount":0,"newContent":"受害者登录 http://a.com ，并保留了登录凭证（Cookie）\n攻击者引诱受害者访问了 http://b.com\nhttp://b.com 发送了一个请求： http://a.com/act=xx 。浏览器会默认携带 http://a.com 的Cookie。\nhttp://a.com 接收到请求后，对请求进行验证，并确认是受害者的凭证，误以为是受害者自己发送的请求。\nhttp://a.com 以受害者的名义执行了act=xx。\n攻击完成，攻击者在受害者不知情的情况下冒充受害者，让 http://a.com 执行了自己定义的操作。\n\n### 几种常见的攻击方式\n\n自动发起 GET 请求的 CSRF\n\n自动发起 POST 请求的 CSRF\n\n引诱用户点击链接的 CSRF\n\n点这里看\n[原文](https://github.com/Advanced-Frontend/Daily-Interview-Question/issues/494)","seeCount":0,"answerUrl":""},{"id":830,"newTitle":"HTTP 中的 keep-alive 有了解吗？它和多路复用的区别","subDes":"","level":3,"viewCount":0,"newContent":"HTTP/1.x keep-alive 与 HTTP/2 多路复用区别：\n\nHTTP/1.x 是基于文本的，只能整体去传；HTTP/2 是基于二进制流的，可以分解为独立的帧，交错发送\n\nHTTP/1.x keep-alive 必须按照请求发送的顺序返回响应；HTTP/2 多路复用不按序响应\n\nHTTP/1.x keep-alive 为了解决队头阻塞，将同一个页面的资源分散到不同域名下，开启了多个 TCP 连接；HTTP/2 同域名下所有通信都在单个连接上完成\n\nHTTP/1.x keep-alive 单个 TCP 连接在同一时刻只能处理一个请求（两个请求的生命周期不能重叠）；HTTP/2 单个 TCP 同一时刻可以发送多个请求和响应\n\n详细解析：[原文](https://github.com/Advanced-Frontend/Daily-Interview-Question/issues/498)","seeCount":0,"answerUrl":""},{"id":829,"newTitle":"介绍下 HTTPS 中间人攻击","subDes":"0","level":3,"viewCount":0,"newContent":"https协议由 http + ssl 协议构成，具体的链接过程可参考 [SSL或TLS握手的概述](https://github.com/lvwxx/blog/issues/3) \n\n### 中间人攻击过程如下：\n\n服务器向客户端发送公钥。\n攻击者截获公钥，保留在自己手上。\n然后攻击者自己生成一个【伪造的】公钥，发给客户端。\n客户端收到伪造的公钥后，生成加密hash值发给服务器。\n攻击者获得加密hash值，用自己的私钥解密获得真秘钥。\n同时生成假的加密hash值，发给服务器。\n服务器用私钥解密获得假秘钥。\n服务器用加秘钥加密传输信息\n防范方法：\n\n服务端在发送浏览器的公钥中加入CA证书，浏览器可以验证CA证书的有效性\n\n[HTTPS原理及握手过程 #1](https://github.com/liuhanqu/fe-interview/issues/1)\n","seeCount":0,"answerUrl":""},{"id":828,"newTitle":"为什么 HTTP1.1 不能实现多路复用（腾讯）","subDes":"","level":2,"viewCount":0,"newContent":"### 解答1\n\nHTTP/1.1 不是二进制传输，而是通过文本进行传输。由于没有流的概念，在使用并行传输（多路复用）传递数据时，接收端在接收到响应后，并不能区分多个响应分别对应的请求，所以无法将多个响应的结果重新进行组装，也就实现不了多路复用。\n\n### 解答2\n\nHTTP1.x是序列和阻塞机制\n\nHTTP 2.0 是多工复用TCP连接，在一个连接里，客户端和浏览器都可以同时发送多个请求或回应，而且不用按照顺序一一对应，这样就避免了@[Zz]队头堵塞@[Zz]。\n\n举例来说，在一个TCP连接里面，服务器同时收到了A请求和B请求，于是先回应A请求，结果发现处理过程非常耗时，于是就发送A请求已经处理好的部分， 接着回应B请求，完成后，再发送A请求剩下的部分。\n\n旧的http1.1是会等 A请求完全处理完后在 处理B请求，会阻塞\n\n另：http1.1已经实现了管道机制：即 在同一个TCP连接里面，客户端可以同时发送多个请求。http 1.0并做不到，所以效率很低\n\n### 解答3\n\n多路复用归功于， HTTP/2 中的 帧（frame）和流（stream）。帧代表着最小的数据单位，每个帧会标识出该帧属于哪个流，流也就是多个帧组成的数据流。就是在一个 TCP 连接中可以存在多条流。\n\n而Http 1.x 并没有这个标识，每次请求都会建立一次HTTP连接，3次握手4次挥手。\n\n\n以上解答来自：\nhttps://github.com/Advanced-Frontend/Daily-Interview-Question/issues/290","seeCount":0,"answerUrl":""},{"id":827,"newTitle":"接口如何防刷","subDes":"","level":2,"viewCount":0,"newContent":"## 解答1\n1：网关控制流量洪峰，对在一个时间段内出现流量异常，可以拒绝请求（参考个人博客文章 https://mp.csdn.net/postedit/81672222）\n\n 2：源ip请求个数限制。对请求来源的ip请求个数做限制 \n\n3：http请求头信息校验；（例如host，User-Agent，Referer）\n\n 4：对用户唯一身份uid进行限制和校验。例如基本的长度，组合方式，甚至有效性进行判断。或者uid具有一定的时效性 \n5：前后端协议采用二进制方式进行交互或者协议采用签名机制 \n6：人机验证，验证码，短信验证码，滑动图片形式，12306形式 \n\n 原文链接：https://blog.csdn.net/timy07/article/details/86467994\n\n\n## 解答2\n\n刷是禁止不了的，只能提高刷的成本\n\n前端：\n\n限制按钮，表单提交次数，防抖\n增加图片验证码，类似12306\n后端：\n\nip，user_agent，referer\n用户唯一标示\n简单验证码\n手机号验证码\napi key，rsa加密认证，ca认证（后端与后端之间调用）\nip黑名单（一般是刷的频率太大，达到ddos水平了或者确认为恶意调用）\n\n\n详解： https://github.com/Advanced-Frontend/Daily-Interview-Question/issues/254","seeCount":0,"answerUrl":""},{"id":826,"newTitle":"谈谈你对 TCP 三次握手和四次挥手的理解","subDes":"0","level":2,"viewCount":0,"newContent":"\n辅助理解：\n\n男：我要挂了哦\n女：等哈，我还要敷面膜\n女：我敷完了，现在可以挂了\n男：我舍不得挂，你挂吧\n女：好吧，我挂了\n男：等了2MSL听见嘟嘟嘟的声音后挂断\n\n---\n\n三次握手之所以是三次是保证client和server均让对方知道自己的接收和发送能力没问题而保证的最小次数。\n\n第一次client => server 只能server判断出client具备发送能力\n第二次 server => client client就可以判断出server具备发送和接受能力。此时client还需让server知道自己接收能力没问题于是就有了第三次\n第三次 client => server 双方均保证了自己的接收和发送能力没有问题\n\n其中，为了保证后续的握手是为了应答上一个握手，每次握手都会带一个标识 seq，后续的ACK都会对这个seq进行加一来进行确认。\n\n详细解析：[TCP 三次握手和四次挥手的理解](https://github.com/Advanced-Frontend/Daily-Interview-Question/issues/15)","seeCount":0,"answerUrl":""},{"id":825,"newTitle":"简单讲解一下http2的多路复用","subDes":"","level":2,"viewCount":0,"newContent":"HTTP2采用二进制格式传输，取代了HTTP1.x的文本格式，二进制格式解析更高效。\n多路复用代替了HTTP1.x的序列和阻塞机制，所有的相同域名请求都通过同一个TCP连接并发完成。在HTTP1.x中，并发多个请求需要多个TCP连接，浏览器为了控制资源会有6-8个TCP连接都限制。\nHTTP2中\n\n同域名下所有通信都在单个连接上完成，消除了因多个 TCP 连接而带来的延时和内存消耗。\n单个连接上可以并行交错的请求和响应，之间互不干扰.\n\n在 HTTP/1 中，每次请求都会建立一次HTTP连接，也就是我们常说的3次握手4次挥手，这个过程在一次请求过程中占用了相当长的时间，即使开启了 Keep-Alive ，解决了多次连接的问题，但是依然有两个效率上的问题：\n\n第一个：串行的文件传输。当请求a文件时，b文件只能等待，等待a连接到服务器、服务器处理文件、服务器返回文件，这三个步骤。我们假设这三步用时都是1秒，那么a文件用时为3秒，b文件传输完成用时为6秒，依此类推。（注：此项计算有一个前提条件，就是浏览器和服务器是单通道传输）\n第二个：连接数过多。我们假设Apache设置了最大并发数为300，因为浏览器限制，浏览器发起的最大请求数为6，也就是服务器能承载的最高并发为50，当第51个人访问时，就需要等待前面某个请求处理完成。\nHTTP/2的多路复用就是为了解决上述的两个性能问题。 在 HTTP/2 中，有两个非常重要的概念，分别是帧（frame）和流（stream）。 帧代表着最小的数据单位，每个帧会标识出该帧属于哪个流，流也就是多个帧组成的数据流。 多路复用，就是在一个 TCP 连接中可以存在多条流。换句话说，也就是可以发送多个请求，对端可以通过帧中的标识知道属于哪个请求。通过这个技术，可以避免 HTTP 旧版本中的队头阻塞问题，极大的提高传输性能。\n\n详细解析：https://github.com/Advanced-Frontend/Daily-Interview-Question/issues/14","seeCount":0,"answerUrl":""},{"id":824,"newTitle":"说下 URL 和 URI 的区别？","subDes":"0","level":2,"viewCount":0,"newContent":"一、URI-统一资源标识符\nURI 全称 Uniform Resource Identifier,指的是统一资源标识符，用来标识唯一的资源。\n\nWeb 上可用的每种资源-HTML 文档、图像、视频片段、程序等都由一个通用资源标识符(即 URI)进行定位。\n\nURI 一般由三部分组成：\n\n1.访问资源的命名机制\n2.存放资源的主机名\n3.资源自身的名称，由路径表示\n看个例子：\n```\nhttp://www.xxx.com/html/html4\n```\n\n这个 URi 是这样的：这是一个通过 HTTP 协议访问的资源，位于\n```\nwww.xxx.com上，通过路径@[Zz]/html/html4@[Zz]访问。\n```\n有的 URI 指向一个资源的内部。这种 URi 以@[Zz]#@[Zz]结束，并跟着一个 anchor 标识符（称为片段标识符）。例如，下面是一个指向 section_2 的 URI：\n```\nhttp://somesite.com/html/top.htm#section_2\n```\n1.1 绝对 URI\nURI 有绝对和相对之分，绝对的 URI 指以 scheme（后面跟着冒号）开头的 URI。前面提到的`http://www.cnn.com`就是绝对的 URI 的一个例子，其它的例子还有`mailto:jeff@javajeff.com`、`news:comp.lang.java.help`和xyz://whatever。你可以把绝对的 URI 看作是以某种方式引用某种资源，而这种方式对标识符出现的环境没有依赖。如果使用文件系统作类比，绝对的 URI 类似于从根目录开始的某个文件的径。\n\n1.2 相对 URI\n相对 URI 不包含任何命名规范信息。它的路径通常指同一台机器上的资源。相对 URI 可能含有相对路径（如，@[Zz]..@[Zz]表示上一层路径），还可能包含片段标识符。比如看下面的例子：\n\n相对 URI：\nsuppliers.html\n它扩展成完全的 URI 就是 \n```\nhttp://www.xxx.com/suppliers.htm\n```\n一个图形的相对 URI\n```\n<img src=@[Zz]../icons/logo.png@[Zz] alt=@[Zz]logo@[Zz]>\n```\n它扩展成完全的 URI 就是\n```\nhttp://www.xxx.com/icons/logo.png\n```\n与绝对的 URI 不同的，相对的 URI 不是以 scheme（后面跟着冒号）开始的 URI。 它的一个例子是articles/articles.html。你可以把相对的 URI 看作是以某种方式引用某种资源，而这种方式依赖于标识符出现的环境。如果用文件系统作类比，相对的 URI 类似于从当前目录开始的文件路径。\n\n1.3 URI 的用途\n在 HTML 中，URI 被用来：\n\n链接到另一个文档或资源\n链接到一个外部样式表或脚本\n在页内包含图形、对象或 applet\n建立图像映射\n提交一个表单\n建立一个框架文档\n引用一个外部参考\n指向一个描述文档的 metadata\n二、URL-统一资源定位器\nURL 全称是 Uniform Resource Locator,指的是统一资源定位器,它是一种具体的 URI，即 URL 可以用来标识一个资源，而且还指明了如何 locate 这个资源。通俗地说，URL 是 Internet 上用来描述资源的字符串，主要用在各种 www 客户端和服务器程序是，特别是著名的 Mosaic。采用 URL 可以用一种统一的格式来描述各种信息资源，包括文件、服务器的地址和目录等。\n\nURL 的第一个部分 http://表示要访问的文件的类型。在网上，这几乎总是使用 http（超文本传输协议，hypertext transfer protocol-用来转换网页的协议）；有时也使用 ftp（文件传输协议，file transfer protocol-用来传输软件和大文件；telnet（远程登录），主要用于远程交谈以及文件调用等，意思是浏览器正在阅读本地盘外的一个文件而不是一个远程计算机。\n\n2.1 URL 组成\nInternet 资源类型（schema）：指出 www 客户程序用来操作的工具。如http://表示 www 服务器，ftp://表示 ftp 服务器，gopher://表示 Gopher 服务器，而new:表示 Newgroup 新闻组。必需的。\n服务器地址（host）：指出 www 网页所在的服务器域名。必需的。\n端口（port）：有时（并非总是这样），对某些资源的访问来说，需给出相应的服务器提供端口。可选的。\n路径（path）：指明服务器上某资源的位置。与端口一样，路径并非总是需要的。可选的。\nURL 地址格式排列为：schema://host:port/path\n\n必须注意：www 上的服务器都是区分大小写的，所以千万要注意正确的 URL 大小写表达形式。\n\n三、URN-统一资源名称\nURN,全称 Uniform Resource Name,指的是统一资源命名,是通过名字来标识资源。比如 `mailto:java-net@java.sun.com`\n\nURN 是 URL 的一种更新形式，统一资源名称(URN, Uniform Resource Name)不依赖于位置，并且有可能减少失效连接的个数。\n\n四、总结\nURI: Uniform Resource Identifier 指的是统一资源标识符\nURL: Uniform Resource Location 指的是统一资源定位符\nURN: Universal Resource Name 指的是统一资源名称\n\nURI 指的是统一资源标识符，用唯一的标识来确定一个资源，它是一种抽象的定义，也就是说，不管使用什么方法来定义，只要能唯一的标识一个资源，就可以称为 URI。它是以某种统一的(标准化的)方式标识资源的简单字符串。\n\nURL 和 URN 是 URI 的子集，URL 可以理解为使用地址来标识资源，URN 可以理解为使用名称来标识资源。\n\nWeb 上地址的基本形式是 URI，它代表统一资源标识符,有两种形式\n\nURL:目前 URI 的最普遍的形式就是无处不在的 URL 或统一资源定位器\nURN:URL 的一种更新形式，统一资源名称不依赖于位置，并且有可能减少失效链接的个数。但是其流行还需要时间，因为它需要更精密软件的支持。\n\n原文地址： https://github.com/lgwebdream/FE-Interview/issues/1201","seeCount":0,"answerUrl":""},{"id":823,"newTitle":"谈谈vuex原理？","subDes":"0","level":3,"viewCount":0,"newContent":"执行Vue.use时会执行vuex的install方法，会往全局混入一个全局的mixin，只有一个属性beforeCreate，它的作用是让每个组件可以访问到this.$store属性。\n\n执行new Vuex.Store时会将传入的配置进行格式化处理，会递归的注册每个module的state、getters、mutation、actions属性，将每个module的getter、action、mutations放入一个对象里，对应的key前面会加上模块名，而state会放入一个有上下级关系的对象里。\n\n\n内部会重写commit和dispatch，再当前模块触发状态变更时会自动在要触发的commit和dispatch前面加上模块名。\n最后会提供一些map开头的语法糖使用。\n\n扩展阅读：\n[聊聊 Redux 和 Vuex 的设计思想](http://bigerfe.com/iq/n480)\n\n[ vuex是什么？怎么使用？哪种功能场景使用它？](http://bigerfe.com/iq/n609\n)\n\n","seeCount":0,"answerUrl":""},{"id":822,"newTitle":"【简单概括】从输入浏览器网址到页面渲染经历了什么？","subDes":"","level":3,"viewCount":0,"newContent":"\n首先浏览器会判断输入的是否是网址，如果是关键字这会调用默认设置的搜索引擎，是网址首先会执行当前页面的beforeonload事件。然后浏览器进程将URL发送给网络进程，发起URL请求。\n\n再发起之前还会尝试去命中缓存，首先是命中强缓存，会依次从service worker、内存、硬盘、push cache里读取缓存，如果有命中就不发起请求。没有命中则正常发起请求，首先是DNS解析将域名解析为IP地址，找到对应的主机。\n\n然后发起请求资源的请求，如果协议是HTTPS，还需要建立TLS连接。发起HTTP请求之前需要建立TCP连接，三次握手确保双方通信的可靠性之后发起HTTP请求，这个时候会尝试去命中协商缓存，如果没命中发送HTTP的请求行、请求头、请求体，等待服务端处理并响应，如果返回的状态码是301/302，则进行重定向，重新发起导航流程。否则根据content-type的类型响应具体的内容，浏览器拿到资源后四次挥手断开TCP连接，如果有开启connection：keep-alive可保持TCP连接，浏览器进入页面渲染阶段。\n\n\n简要说明下浏览器拿到响应数据后的渲染过程？\n\n整个过程也可以称之为渲染流水线：\n\n\n首先是构建DOM树，将标签解析为一个个的Token，依次压入弹出token栈确认标签的父子关系，构建为一颗树结构。\n\n\n接下来是构建CSSOM，将link标签、style标签、内联的css解析为styleSheet，将属性标准化，如将em转为px、颜色关键字转为rgb值、属性关键字转为具体数值，通过继承、层叠规则遍历DOM树匹配对应css属性，让每个DOM拥有各自对应的样式属性，在浏览器的elelemt-computed里可以看到。\n\n\n接下来将CSSOM和DOM进行属性的计算，重新计算为一颗只包括可见元素的布局树，隐藏的元素不参与布局计算。\n\n\n接下来是分层，例如绝对定位的元素能遮罩其他元素，是因为它们都不在一个图层，需要将布局树再次计算为一颗包含了上下级关系的图层树。\n\n\n再接下来是绘制，根据图层关系逐步绘制，先绘制底部再绘制上层，按照顺序组成一个待绘制列表。\n\n\n不过绘制也不是整个页面全部都绘制，优先是绘制浏览器视口以内的图层，所以还有一个分块的步骤，将图层划分更小的块，先绘制视口内的小块。\n\n\n接下来是光栅化，使用GPU将当前视口的图块渲染为位图，将其保存到GPU内存里。\n\n\n最后光栅化完毕，通知浏览器进程，将GPU内存里的位图显示在屏幕上。\n\n\n重排：通过JS或CSS更改了元素的几何位置，触发重排，从布局阶段开始重新渲染整个流水线。\n重绘：通过JS更改元素外观，触发重绘，从绘制阶段开始渲染流水线，省去了布局和分层阶段。\n合成：不更改布局和外观属性，例如用transform实现动画，直接在合成线程完成动画操作，CSS的will-change属性，提前告知浏览器需要进行CSS动画，为其单独准备一个独立的层，之后通过合成线程完成动画。\n\n\n原文在这里：https://juejin.cn/post/6945025017834897422","seeCount":0,"answerUrl":""},{"id":821,"newTitle":"如何与服务器保持长连接？","subDes":"","level":2,"viewCount":0,"newContent":"ajax轮询： ajax每隔一段时间向服务端发起请求，保持数据的同步。缺点是效率低，资源浪费。\n\nlong poll长轮询： 请求头的connection需要设置为keep-alive，客户端发送请求后，如果没有数据返回，服务端将这个请求挂起放入队列，直到有数据返回，客户端再次发起请求，以此轮询。优点是能减少无效的网络传输；缺点是无法处理高并发的场景。\n\niframe长连接：\n在网页上嵌入一个iframe标签，它的src指向一个长连接请求。优点是消息传输及时；缺点是消耗服务器资源。\n\nwebsocket： 双向通信，只需要连接一次，就可以相互传输数据，适合适时通讯，数据适时更新等场景。websoket协议与http协议没有关系，它是一个建立在tcp协议上的全新协议，为了兼容http握手规范，在握手阶段依然使用http协议，握手完成之后，数据通过tcp进行传输。优点是双向通信，不存在跨域限制，只用建立一次连接；缺点是长连接受网络限制比较大，需要处理好重连，只支持ie10以上版本。\n\nwebsocket与http的区别：\n*  url以ws：或wss：开头。\n*  状态码是101。\n*   请求头和响应头的connection的值是upgrade，表示协议升级。\n*  请求头和响应头会有sec-websoket字段。\n\n\n原文在这里：https://juejin.cn/post/6945025017834897422\n","seeCount":0,"answerUrl":""},{"id":820,"newTitle":"defineProterty的属性值有哪些","subDes":"","level":1,"viewCount":0,"newContent":"* configurable 是否可以删除属性和属性描述\n*  enumerable 才能出现在对象枚举中\n*  value 初始值\n*  writable 是否能被赋值运算符改变\n*  get\n*  set\n\n扩展链接：\nhttps://www.jianshu.com/p/02e985bd0ea8","seeCount":0,"answerUrl":""},{"id":819,"newTitle":"请说下VUE diff算法","subDes":"","level":3,"viewCount":0,"newContent":"![alt](https://p1-jj.byteimg.com/tos-cn-i-t2oaga2asx/gold-user-assets/2019/8/20/16caf6f26aa2d5eb~tplv-t2oaga2asx-watermark.awebp)\n![alt](https://p1-jj.byteimg.com/tos-cn-i-t2oaga2asx/gold-user-assets/2019/8/21/16cb48188c5ed824~tplv-t2oaga2asx-watermark.awebp)\n![alt](https://p1-jj.byteimg.com/tos-cn-i-t2oaga2asx/gold-user-assets/2019/8/22/16cb4f706d65e80c~tplv-t2oaga2asx-watermark.awebp)\n![alt](https://p1-jj.byteimg.com/tos-cn-i-t2oaga2asx/gold-user-assets/2019/8/22/16cb4f897d9a8ed9~tplv-t2oaga2asx-watermark.awebp)\n\n![alt](https://p1-jj.byteimg.com/tos-cn-i-t2oaga2asx/gold-user-assets/2019/8/22/16cb9a8c163e2821~tplv-t2oaga2asx-watermark.awebp)\n\n![alt](https://p1-jj.byteimg.com/tos-cn-i-t2oaga2asx/gold-user-assets/2019/8/22/16cb9a9246bd8470~tplv-t2oaga2asx-watermark.awebp)\n\n更多详细解析请看原文：\n[一起搞明白令人头疼的diff算法](https://juejin.cn/post/6844903921408802829)","seeCount":0,"answerUrl":""},{"id":818,"newTitle":"VUE watch和computed的区别","subDes":"","level":2,"viewCount":0,"newContent":"* 使用场景\n* computed有缓存\n* 都是vue watcher类\n\n详细解析：\n[搞懂computed和watch原理，减少使用场景思考时间](https://juejin.cn/post/6844903926819454983)","seeCount":0,"answerUrl":""},{"id":817,"newTitle":"手写用 ES6proxy 如何实现 arr[-1] 的访问","subDes":"","level":3,"viewCount":0,"newContent":"参考解析：\n\n```\nconst proxyArray = arr => {\n    const length = arr.length;\n    return new Proxy(arr, {\n        get(target, key) {\n            key = +key;\n            while (key < 0) {\n                key += length;\n            }\n            return target[key];\n        }\n    })\n};\nvar a = proxyArray([1, 2, 3, 4, 5, 6, 7, 8, 9]);\nconsole.log(a[1]);  // 2\nconsole.log(a[-10]);  // 9\nconsole.log(a[-20]);  // 8\n```\n\n更详细解析：\nhttps://github.com/lgwebdream/FE-Interview/issues/36","seeCount":0,"answerUrl":""},{"id":816,"newTitle":"node 中间层怎样做的请求合并转发?","subDes":"0","level":1,"viewCount":0,"newContent":"\n1） 什么是中间层\n\n就是前端---请求---> nodejs ----请求---->后端 ----响应--->nodejs--数据处理---响应---->前端。这么一个流程，这个流程的好处就是当业务逻辑过多，或者业务需求在不断变更的时候，前端不需要过多当去改变业务逻辑，与后端低耦合。前端即显示，渲染。后端获取和存储数据。中间层处理数据结构，返回给前端可用可渲染的数据结构。\nnodejs是起中间层的作用，即根据客户端不同请求来做相应的处理或渲染页面，处理时可以是把获取的数据做简单的处理交由底层java那边做真正的数据持久化或数据更新，也可以是从底层获取数据做简单的处理返回给客户端。\n通常我们把Web领域分为客户端和服务端，也就是前端和后端，这里的后端就包含了网关，静态资源，接口，缓存，数据库等。而中间层呢，就是在后端这里再抽离一层出来，在业务上处理和客户端衔接更紧密的部分，比如页面渲染（SSR），数据聚合，接口转发等等。\n以SSR来说，在服务端将页面渲染好，可以加快用户的首屏加载速度，避免请求时白屏，还有利于网站做SEO，他的好处是比较好理解的。\n\n2）中间层可以做的事情\n\n代理：在开发环境下，我们可以利用代理来，解决最常见的跨域问题；在线上环境下，我们可以利用代理，转发请求到多个服务端。\n\n缓存：缓存其实是更靠近前端的需求，用户的动作触发数据的更新，node中间层可以直接处理一部分缓存需求。\n限流：node中间层，可以针对接口或者路由做响应的限流。\n日志：相比其他服务端语言，node中间层的日志记录，能更方便快捷的定位问题（是在浏览器端还是服务端）。\n监控：擅长高并发的请求处理，做监控也是合适的选项。\n鉴权：有一个中间层去鉴权，也是一种单一职责的实现。\n路由：前端更需要掌握页面路由的权限和逻辑。\n服务端渲染：node中间层的解决方案更灵活，比如SSR、模板直出、利用一些JS库做预渲染等等。\n\n3）node转发API（node中间层）的优势\n\n可以在中间层把java|php的数据，处理成对前端更友好的格式\n可以解决前端的跨域问题，因为服务器端的请求是不涉及跨域的，跨域是浏览器的同源策略导致的\n可以将多个请求在通过中间层合并，减少前端的请求\n\n4）如何做请求合并转发\n\n使用express中间件multifetch可以将请求批量合并\n使用express+http-proxy-middleware实现接口代理转发\n\n5）不使用用第三方模块手动实现一个nodejs代理服务器，实现请求合并转发\n\n---\n\n以上为部分解析，更详细解析请查看：\nhttps://github.com/lgwebdream/FE-Interview/issues/28\n\n","seeCount":0,"answerUrl":""},{"id":815,"newTitle":"实现 add(1)(2)(3)","subDes":"0","level":2,"viewCount":9809,"newContent":"考点：函数柯里化\n\n函数柯里化概念： 柯里化（Currying）是把接受多个参数的函数转变为接受一个单一参数的函数，并且返回接受余下的参数且返回结果的新函数的技术。\n\n## 粗暴版本\n```\nfunction add (a) {\n\treturn function (b) {\n\t\treturn function (c) {\n\t\t    return a + b + c;\n\t\t}\n\t}\n}\nconsole.log(add(1)(2)(3)); // 6\n```\n## 柯里化解决方案\n\n参数长度固定\n\n```\nconst curry = (fn) =>\n(judge = (...args) =>\n    args.length === fn.length\n    ? fn(...args)\n    : (...arg) => judge(...args, ...arg));\nconst add = (a, b, c) => a + b + c;\nconst curryAdd = curry(add);\nconsole.log(curryAdd(1)(2)(3)); // 6\nconsole.log(curryAdd(1, 2)(3)); // 6\nconsole.log(curryAdd(1)(2, 3)); // 6\n```\n\n参数长度不固定\n\n```\nfunction add (...args) {\n    //求和\n    return args.reduce((a, b) => a + b)\n}\n\nfunction currying (fn) {\n    let args = []\n    return function temp (...newArgs) {\n        if (newArgs.length) {\n            args = [\n                ...args,\n                ...newArgs\n            ]\n            return temp\n        } else {\n            let val = fn.apply(this, args)\n            args = [] //保证再次调用时清空\n            return val\n        }\n    }\n}\n\nlet addCurry = currying(add)\nconsole.log(addCurry(1)(2)(3)(4, 5)())  //15\nconsole.log(addCurry(1)(2)(3, 4, 5)())  //15\nconsole.log(addCurry(1)(2, 3, 4, 5)())  //15\n```\n\n以上解析来自 ---》 \n查看更详细解析--》\nhttps://github.com/lgwebdream/FE-Interview/issues/21\n","seeCount":0,"answerUrl":""},{"id":814,"newTitle":"说说TCP 滑动窗口？","subDes":"","level":2,"viewCount":112,"newContent":"在 TCP 链接中，对于发送端和接收端而言，TCP 需要把发送的数据放到发送缓存区, 将接收的数据放到接收缓存区。而经常会存在发送端发送过多，而接收端无法消化的情况，所以就需要流量控制，就是在通过接收缓存区的大小，控制发送端的发送。如果对方的接收缓存区满了，就不能再继续发送了。而这种流量控制的过程就需要在发送端维护一个发送窗口，在接收端维持一个接收窗口。\nTCP 滑动窗口分为两种: 发送窗口和接收窗口。\n\n参考资料：\nhttps://juejin.cn/post/6844904070889603085#heading-38","seeCount":0,"answerUrl":""},{"id":813,"newTitle":"简单实现async/await中的async函数","subDes":"async 函数的实现原理，就是将 Generator 函数和自动执行器，包装在一个函数里\n\n```\nfunction spawn(genF) {\n    return new Promise(function(resolve, reject) {\n        const gen = genF();\n        function step(nextF) {\n            let next;\n            try {\n                next = nextF();\n            } catch (e) {\n                return reject(e);\n            }\n            if (next.done) {\n                return resolve(next.value);\n            }\n            Promise.resolve(next.value).then(\n                function(v) {\n                    step(function() {\n                        return gen.next(v);\n                    });\n                },\n                function(e) {\n                    step(function() {\n                        return gen.throw(e);\n                    });\n                }\n            );\n        }\n        step(function() {\n            return gen.next(undefined);\n        });\n    });\n}\n\n```\n原文：\nhttps://github.com/airuikun/technology-blog/issues/14","level":2,"viewCount":0,"newContent":"async 函数的实现原理，就是将 Generator 函数和自动执行器，包装在一个函数里\n\n```\nfunction spawn(genF) {\n    return new Promise(function(resolve, reject) {\n        const gen = genF();\n        function step(nextF) {\n            let next;\n            try {\n                next = nextF();\n            } catch (e) {\n                return reject(e);\n            }\n            if (next.done) {\n                return resolve(next.value);\n            }\n            Promise.resolve(next.value).then(\n                function(v) {\n                    step(function() {\n                        return gen.next(v);\n                    });\n                },\n                function(e) {\n                    step(function() {\n                        return gen.throw(e);\n                    });\n                }\n            );\n        }\n        step(function() {\n            return gen.next(undefined);\n        });\n    });\n}\n\n```\n原文：\nhttps://github.com/airuikun/technology-blog/issues/14","seeCount":0,"answerUrl":""},{"id":795,"newTitle":"平时怎么学习(淘系）","subDes":"0","level":1,"viewCount":0,"newContent":"这是一个主观题，但是非常值得深思。\n\n如果只说看看某某社区文章和书籍学习可能不是特别好的答案。\n\n要有自己的特点。。。\n\n我的回答（仅供参考）：\n\n首先有自己个学习节奏，有自己的学习方法，我有一个自己的知识体系，知识体系会不断的丰富和更新。具体细节的内容会参考一些社区内的文章进行理解，然后针对性的输出一些技术文章，进行沉淀。\n\n另外会参考一些国内知名的网站，看下他们的网页源码，看下他们的规范和应用。\n\n再就是多思考，多总结，多输出。\n","seeCount":0,"answerUrl":""},{"id":794,"newTitle":"PWA原理，讲下核心点（淘系）","subDes":"","level":2,"viewCount":0,"newContent":"参考文章，文章较长，需要自我总结和提炼\n\nhttps://blog.csdn.net/weixin_33692284/article/details/88027509","seeCount":0,"answerUrl":""},{"id":793,"newTitle":"简述TCP连接的过程（淘系）","subDes":"","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://blog.csdn.net/qq_42196196/article/details/81042376"},{"id":792,"newTitle":"怎么用栈模拟队列，队列模拟栈，有思路吗（淘系）","subDes":"","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://blog.csdn.net/sddh1988/article/details/62422792"},{"id":791,"newTitle":"快排原理以及时间复杂度，为什么（淘系前端）","subDes":"0","level":3,"viewCount":0,"newContent":"\n\n快排和归排的复杂度都是O(n*log n)，为什么都用快排而不用归排？\n\n看了《算法图解》之后，大致理解了是什么原因，真正的原因是：不可描述的常量导致使用快排而不是归排。\n\n好了，真正的解释是这样的：\n\n算法的每一步实际上都需要一个固定时间量，被称为常量。我们平时考虑时间复杂度的时候并不考虑常量的影响，但有时候常量的影响不可忽略，比如在这个问题上。但是大多数时候考虑复杂度的时候，可能还是不需要考虑常量的影响的，嗯，我觉得是……由于算法的每一步都有一个常量，而快排的常量比归排小，因此虽然两者的复杂度相同，但是快排更快一些。\n\n那第二个问题就来了，归排的复杂度一直都是O(n*log n)，而快排的平均复杂度才是O(n*log n)，最坏情况下快排的复杂度可以达到O(n^2)，为什不怕快排的时候是最坏的情况呢？主要是因为绝大多数情况下，快排遇到的都是平均情况，也就是最佳情况，只有极个别的时候会是最坏情况，因此往往不考虑这种糟糕的情况。\n\n\n---\n\n作者：吉大秦少游\n原文链接：https://blog.csdn.net/zhanshen112/article/details/85211505\n\n\n其他参考：https://blog.csdn.net/qq_20746945/article/details/89378662","seeCount":0,"answerUrl":"https://blog.csdn.net/qq_20746945/article/details/89378662 "},{"id":790,"newTitle":"mvvm与mvc的区别(淘系前端）","subDes":"","level":1,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://juejin.im/post/5b3a3a44f265da630e27a7e6 https://www.cnblogs.com/mlw1814011067/p/10400240.html "},{"id":789,"newTitle":"你认为flutter与react native区别于在哪（淘系前端）","subDes":"","level":1,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://blog.csdn.net/lyc00net/article/details/88250759"},{"id":788,"newTitle":"addEventListener第三个参数是什么，还可以是别的吗（淘系前端）","subDes":"","level":1,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://www.jianshu.com/p/bad857d649f2 https://developer.mozilla.org/zh-CN/docs/Web/API/EventTarget/addEventListener\n"},{"id":775,"newTitle":"饿了么 - 说下 koa 的中间件机制","subDes":"说下 koa 的中间件机制，解决了什么问题，如何实现的。","level":2,"viewCount":0,"newContent":"个人心得：\n\n要说出自己的理解，越详细越好。\n\n所以在你理解了这个技术后，要形成自己的理解和表达，如何用简单的语言描述清楚（这个很重要），面试官能从你的表达中感受到你到底是不是真的懂，是否看过源码。\n\n详解可以看这个：\nhttps://juejin.im/post/5c7decbbe51d454a7c5e8474","seeCount":0,"answerUrl":""},{"id":774,"newTitle":"说下单点登录-新东方","subDes":"如题，请介绍下对单点登录的理解","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://juejin.im/post/5b73c71fe51d45666016655a"},{"id":771,"newTitle":"pureComponent和FunctionComponent区别（有赞）","subDes":"","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://juejin.im/post/5d3ef3646fb9a06b1b1999fd https://www.cnblogs.com/ajaxkong/p/11419061.html"},{"id":770,"newTitle":"项目中如何处理安全问题（阿里）","subDes":"0","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://www.cnblogs.com/aiandbigdata/p/10057659.html"},{"id":769,"newTitle":"居中为什么要使用transform（为什么不使用marginLeft/Top）（阿里）","subDes":"0","level":2,"viewCount":0,"newContent":"transform 属于合成属性（composite property），对合成属性进行 transition/animation 动画将会创建一个合成层（composite layer），这使得被动画元素在一个独立的层中进行动画。通常情况下，浏览器会将一个层的内容先绘制进一个位图中，然后再作为纹理（texture）上传到 GPU，只要该层的内容不发生改变，就没必要进行重绘（repaint），浏览器会通过重新复合（recomposite）来形成一个新的帧。\n\ntop/left属于布局属性，该属性的变化会导致重排（reflow/relayout），所谓重排即指对这些节点以及受这些节点影响的其它节点，进行CSS计算->布局->重绘过程，浏览器需要为整个层进行重绘并重新上传到 GPU，造成了极大的性能开销。\n\n---\n\n详解：\n\nhttps://juejin.im/post/5c32b0fb6fb9a049ac7950d9\n\nhttps://blog.csdn.net/callmeCassie/article/details/89290945\n\nhttps://www.cnblogs.com/accordion/p/4593576.html\n\nhttps://www.zhihu.com/question/33629083\n\n具体应用：\n\nhttps://www.jianshu.com/p/17e289fcf467\n\n\n","seeCount":0,"answerUrl":""},{"id":768,"newTitle":"如何解决移动端适配1px的问题（阿里）","subDes":"","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://juejin.im/post/5d19b729f265da1bb2774865"},{"id":767,"newTitle":"Promise.all 并发限制 同时只能有5个请求（瓜子）","subDes":"","level":3,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://segmentfault.com/a/1190000016389127 "},{"id":765,"newTitle":"src和 href 的区别（京东）","subDes":"0","level":1,"viewCount":0,"newContent":"href标识超文本引用，用在link和a等元素上，href是引用和页面关联，是在当前元素和引用资源之间建立联系。\n\nsrc表示引用资源，表示替换当前元素，用在img，script，iframe上，src是页面内容不可缺少的一部分。\n\nsrc是source的缩写，是指向外部资源的位置，指向的内部会迁入到文档中当前标签所在的位置；在请求src资源时会将其指向的资源下载并应用到当前文档中，例如js脚本，img图片和frame等元素。\n\n```\n<script src=@[Zz]js.js@[Zz]></script>\n```\n当浏览器解析到这一句的时候会暂停其他资源的下载和处理，直至将该资源加载，编译，执行完毕，图片和框架等元素也是如此，类似于该元素所指向的资源嵌套如当前标签内，这也是为什么要把js饭再底部而不是头部。\n\n---\n\n原文链接\nhttps://blog.csdn.net/binlety/article/details/81448195","seeCount":0,"answerUrl":""},{"id":764,"newTitle":"react-router里的<Link>标签和<a>标签有什么区别（滴滴）","subDes":"0","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://blog.csdn.net/sinat_17775997/article/details/66967854"},{"id":763,"newTitle":"react-router怎么实现路由切换（滴滴）","subDes":"","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://blog.csdn.net/leviscar/article/details/81878677"},{"id":762,"newTitle":"React组件事件代理的原理（网易）","subDes":"","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://juejin.im/post/5d7678b06fb9a06b2b47a03c"},{"id":761,"newTitle":"请介绍Promise，异常捕获（网易）","subDes":"0","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://blog.csdn.net/billll/article/details/74081426 https://blog.csdn.net/kabike/article/details/101460637"},{"id":760,"newTitle":"介绍下粘性布局（sticky）（网易）","subDes":"0","level":1,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://www.cnblogs.com/s1nker/p/4835079.html"},{"id":759,"newTitle":"http1.1时如何复用tcp连接（网易）","subDes":"0","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://www.jianshu.com/p/52d86558ca57"},{"id":758,"newTitle":"观察者和订阅-发布的区别，各自用在哪里（网易）","subDes":"","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://juejin.im/post/5a14e9edf265da4312808d86"},{"id":757,"newTitle":"文件上传如何做断点续传（网易）","subDes":"","level":1,"viewCount":0,"newContent":"---\n详细解析\nhttps://juejin.im/post/5da14778f265da5bb628e590#heading-17","seeCount":0,"answerUrl":"https://juejin.im/post/5da14778f265da5bb628e590#heading-17"},{"id":755,"newTitle":"使用过的koa2中间件（阿里）","subDes":"","level":1,"viewCount":0,"newContent":"```\nkoa-router\nkoa-bodyparser\nkoa-views\nkoa-static\nkoa-jwt\nkoa-helmet\nkoa-convert\n```\n\n--\n详解：https://www.jianshu.com/p/c1e0ca3f9764","seeCount":0,"answerUrl":"https://www.jianshu.com/p/c1e0ca3f9764"},{"id":754,"newTitle":"master挂了的话pm2怎么处理（阿里）","subDes":"","level":3,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://www.cnblogs.com/Yanss/p/10168359.html"},{"id":753,"newTitle":"介绍SSL和TLS（寺库）","subDes":"","level":3,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://www.cnblogs.com/susanhonly/p/7489532.html"},{"id":752,"newTitle":"说说网络的五层模型（寺库）","subDes":"0","level":2,"viewCount":0,"newContent":"应用层\n\n传输层\n\n网络层\n\n数据链路层\n\n物理层\n--\n参考解析：\n\nhttps://blog.csdn.net/liuchengzimozigreat/article/details/100169829\n\nhttps://www.jianshu.com/p/ad820267da1c","seeCount":0,"answerUrl":""},{"id":751,"newTitle":"RN的原理，为什么可以同时在安卓和IOS端运行（寺库）","subDes":"0","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://www.cnblogs.com/guyuehuanhuan/p/6847979.html"},{"id":750,"newTitle":"promise如何实现then处理（宝宝树）","subDes":"","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://zhuanlan.zhihu.com/p/32894040"},{"id":749,"newTitle":"介绍MVP怎么组织（宝宝树）","subDes":"0","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://blog.csdn.net/weixin_38031122/article/details/79287744 https://www.cnblogs.com/Decmber/p/5279028.html"},{"id":748,"newTitle":"node文件查找优先级（宝宝树）","subDes":"","level":1,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://cloud.tencent.com/developer/article/1005768"},{"id":747,"newTitle":"npm2和npm3 有什么区别(宝宝树)","subDes":"","level":2,"viewCount":0,"newContent":"如果要我简单来说的话，就是 npm3 和 npm2 对于依赖的处理不一样了\n\nnpm2所有项目依赖是嵌套关系，而npm3为了改进嵌套过多、套路过深的情况，会将所有依赖放在第二层依赖中（所有依赖只嵌套一次，彼此平行，也就是平铺的结构）\n\n---\n参考解析：https://www.jianshu.com/p/69ba32550c08","seeCount":0,"answerUrl":""},{"id":736,"newTitle":"什么时候使用“git rebase”代替“git merge”？","subDes":"0","level":1,"viewCount":0,"newContent":"这两个命令都是把修改从一个分支集成到另一个分支上，它们只是以非常不同的方式进行。\n\n\n考虑一下场景，在合并和变基前：\n\n```\nA <- B <- C    [master]\n\n^\n\n \n  D <- E       [branch]\n\n ```\n\n在 git merge master 之后：\n\n \n```\nA <- B <- C\n\n^         ^\n\n          \n\n  D <- E <- F\n\n ```\n\n在 git rebase master 之后：\n\n \n```\nA <- B <- C <- D <- E\n\n ```\n\n使用变基时，意味着使用另一个分支作为集成修改的新基础。\n\n \n\n何时使用：\n\n \n\n如果你对修改不够果断，请使用合并操作。\n\n根据你希望的历史记录的样子，而选择使用变基或合并操作。\n\n \n\n更多需要考虑的因素：\n\n \n\n分支是否与团队外部的开发人员共享修改（如开源、公开项目）？如果是这样，请不要使用变基操作。变基会破坏分支，除非他们使用 git pull --rebase，否则这些开发人员将会得到损坏的或不一致的仓库。\n\n你的开发团队技术是否足够娴熟？变基是一种破坏性操作。这意味着，如果你没有正确使用它，你可能会丢失提交，并且/或者会破坏其他开发者仓库的一致性。\n\n分支本身是否代表有用的信息？一些团队使用功能分支（branch-per-feature）模式，每个分支代表一个功能（或错误修复，或子功能等）。在此模式中，分支有助于识别相关提交的集合。在每个开发人员分支（branch-per-developer）模式中，分支本身不会传达任何其他信息（提交信息已有作者）。则在这种模式下，变基不会有任何破坏。\n\n是否无论如何都要还原合并？恢复（如在撤销中）变基，是相当困难的，并且/或者在变基中存在冲突时，是不可能完成的。如果你考虑到日后可能需要恢复，请使用合并操作。","seeCount":0,"answerUrl":""},{"id":735,"newTitle":"告诉我 Git 中 HEAD、工作树和索引之间的区别？","subDes":"","level":1,"viewCount":0,"newContent":"该工作树/工作目录/工作空间是你看到和编辑的（源）文件的目录树。\n\n该索引/中转区（staging area）是个在 /.git/index，单一的、庞大的二进制文件，该文件列出了当前分支中所有文件的 SHA1 检验和、时间戳和文件名，它不是个带有文件副本的目录。\n\nHEAD是当前检出分支的最后一次提交的引用/指针。\n\n\n来源： https://stackoverflow.com/questions/3689838/whats-the-difference-between-head-working-tree-and-index-in-git\n\n ","seeCount":0,"answerUrl":""},{"id":734,"newTitle":"解释 Forking 工作流程的优点","subDes":"","level":1,"viewCount":0,"newContent":"Forking 工作流程 与其他流行的 Git 工作流程有着根本的区别。它不是用单个服务端仓库充当“中央”代码库，而是为每个开发者提供自己的服务端仓库。Forking 工作流程最常用于公共开源项目中。\n\n \nForking 工作流程的主要优点是可以汇集提交贡献，又无需每个开发者提交到一个中央仓库中，从而实现干净的项目历史记录。开发者可以推送（push）代码到自己的服务端仓库，而只有项目维护人员才能直接推送（push）代码到官方仓库中。\n\n \n当开发者准备发布本地提交时，他们的提交会推送到自己的公共仓库中，而不是官方仓库。然后他们向主仓库提交请求拉取（pull request），这会告知项目维护人员有可以集成的更新。\n\n ---\n来源：https://www.atlassian.com/git/tutorials/comparing-workflows/forking-workflow","seeCount":0,"answerUrl":""},{"id":733,"newTitle":"“git pull”和“git fetch”之间有什么区别？   ","subDes":"","level":1,"viewCount":0,"newContent":"简单来说，git pull 是 git fetch   git merge。\n\n当你使用 pull，Git 会试着自动为你完成工作。它是上下文（工作环境）敏感的，所以 Git 会把所有拉取的提交合并到你当前处理的分支中。pull 则是 自动合并提交而没有让你复查的过程。如果你没有细心管理你的分支，你可能会频繁遇到冲突。\n\n\n当你 fetch，Git 会收集目标分支中的所有不存在的提交，并将这些提交存储到本地仓库中。但Git 不会把这些提交合并到当前分支中。这种处理逻辑在当你需要保持仓库更新，在更新文件时又希望处理可能中断的事情时，这将非常实用。而将提交合并到主分支中，则该使用 merge。","seeCount":0,"answerUrl":""},{"id":732,"newTitle":"“拉取请求（pull request）”和“分支（branch）”之间有什么区别？","subDes":"","level":1,"viewCount":0,"newContent":"分支（branch） 是代码的一个独立版本。\n\n\n拉取请求（pull request） 是当有人用仓库，建立了自己的分支，做了些修改并合并到该分支（把自己修改应用到别人的代码仓库）。\n\n\n \n\n来源： https://stackoverflow.com/questions/19059838/whats-the-difference-between-a-pull-request-and-a-branch","seeCount":0,"answerUrl":""},{"id":731,"newTitle":"什么是 Git 复刻（fork）？复刻（fork）、分支（branch）和克隆（clone）之间有什么区别？","subDes":"","level":1,"viewCount":0,"newContent":"复刻（fork） 是对存储仓库（repository）进行的远程的、服务器端的拷贝，从源头上就有所区别。复刻实际上不是 Git 的范畴。它更像是个政治/社会概念。\n\n\n克隆（clone） 不是复刻，克隆是个对某个远程仓库的本地拷贝。克隆时，实际上是拷贝整个源存储仓库，包括所有历史记录和分支。\n\n\n分支（branch） 是一种机制，用于处理单一存储仓库中的变更，并最终目的是用于与其他部分代码合并。\n\n \n\n来源：https://stackoverflow.com/questions/3329943/git-branch-fork-fetch-merge-rebase-and-clone-what-are-the-differences/","seeCount":0,"answerUrl":""},{"id":730,"newTitle":"使用过git cherry-pick，有什么作用？","subDes":"0","level":1,"viewCount":0,"newContent":"git cherry-pick可以理解为”挑拣”提交，它会获取某一个分支的单笔提交，并作为一个新的提交引入到你当前分支上。 当我们需要在本地合入其他分支的提交时，如果我们不想对整个分支进行合并，而是只想将某一次提交合入到本地当前分支上，那么就要使用git cherry-pick了。\n\n用法\n\n```\ngit cherry-pick [<options>] <commit-ish>...\n\n常用options:\n    --quit                退出当前的chery-pick序列\n    --continue            继续当前的chery-pick序列\n    --abort               取消当前的chery-pick序列，恢复当前分支\n    -n, --no-commit       不自动提交\n    -e, --edit            编辑提交信息\n```\n\n举个例子\n\n```\n在branch1上查看commit的id,例如是8d63abdcf7c331a5c29e392fa1fa85a2161cbdf4,\n然后切换到master分支,执行cherrt-pick\n\ngit checkout master\ngit cherry-pick 8d63abdcf7c331a5c29e392fa1fa85a2161cbdf4\n```\n\n","seeCount":0,"answerUrl":""},{"id":729,"newTitle":" git跟其他版本控制器有啥区别？","subDes":"","level":1,"viewCount":0,"newContent":"GIT是分布式版本控制系统，其他类似于SVN是集中式版本控制系统。\n分布式区别于集中式在于：每个节点的地位都是平等，拥有自己的版本库，在没有网络的情况下，对工作空间内代码的修改可以提交到本地仓库，此时的本地仓库相当于集中式的远程仓库，可以基于本地仓库进行提交、撤销等常规操作，从而方便日常开发。\n\n---\n\n来源链接：https://blog.csdn.net/nobody_1/article/details/88956315","seeCount":0,"answerUrl":""},{"id":728,"newTitle":"我们在本地工程常会修改一些配置文件，这些文件不需要被提交，而我们又不想每次执行git status时都让这些文件显示出来，我们该如何操作？","subDes":"0","level":1,"viewCount":0,"newContent":"首先利用命令touch .gitignore新建文件\n\n```\n$ touch .gitignore\n```\n\n然后往文件中添加需要忽略哪些文件夹下的什么类型的文件\n\n\n\n```\n$ vim .gitignore\n$ cat .gitignore\n/target/class\n.settings\n.imp\n*.ini\n```\n\n注意：忽略/target/class文件夹下所有后缀名为.settings，.imp的文件，忽略所有后缀名为.ini的文件。\n\n来源：\nhttps://blog.csdn.net/nobody_1/article/details/88956315\n\n","seeCount":0,"answerUrl":""},{"id":727,"newTitle":"如何把本地仓库的内容推向一个空的远程仓库？","subDes":"0","level":1,"viewCount":0,"newContent":"首先确保本地仓库与远程之间是连同的。如果提交失败，则需要进行下面的命令进行连通：\n\n```\ngit remote add origin XXXX\n```\n\n注意：XXXX是你的远程仓库地址。\n\n如果是第一次推送，则进行下面命令：\n\n```\ngit push -u origin master\n```\n\n注意：-u 是指定origin为默认主分支\n\n之后的提交，只需要下面的命令：\n\n```\ngit push origin master\n```\n\n来源：\nhttps://blog.csdn.net/nobody_1/article/details/88956315","seeCount":0,"answerUrl":""},{"id":726,"newTitle":"提交时发生冲突，你能解释冲突是如何产生的吗？你是如何解决的？","subDes":"0","level":1,"viewCount":0,"newContent":"开发过程中，我们都有自己的特性分支，所以冲突发生的并不多，但也碰到过。诸如公共类的公共方法，我和别人同时修改同一个文件，他提交后我再提交就会报冲突的错误。\n\n发生冲突，在IDE里面一般都是对比本地文件和远程分支的文件，然后把远程分支上文件的内容手工修改到本地文件，然后再提交冲突的文件使其保证与远程分支的文件一致，这样才会消除冲突，然后再提交自己修改的部分。特别要注意下，修改本地冲突文件使其与远程仓库的文件保持一致后，需要提交后才能消除冲突，否则无法继续提交。必要时可与同事交流，消除冲突。\n发生冲突，也可以使用命令。\n\n* 通过git stash命令，把工作区的修改提交到栈区，目的是保存工作区的修改；\n* 通过git pull命令，拉取远程分支上的代码并合并到本地分支，目的是消除冲突；\n* 通过git stash pop命令，把保存在栈区的修改部分合并到最新的工作空间中；\n\n\n\n---\n\n来源：https://blog.csdn.net/nobody_1/article/details/88956315\n\n----\n\ngit stash  详细介绍\n\nhttps://www.cnblogs.com/zndxall/archive/2018/09/04/9586088.html","seeCount":0,"answerUrl":"111"},{"id":725,"newTitle":"列举工作中常用的几个git命令？","subDes":"0","level":1,"viewCount":0,"newContent":"新增文件的命令：git add file或者git add .\n\n提交文件的命令：git commit –m或者git commit –a\n\n查看工作区状况：git status –s\n\n拉取合并远程分支的操作：git fetch/git merge或者git pull\n\n查看提交记录命令：git reflog\n\n————————————————\n\n来源：https://blog.csdn.net/nobody_1/article/details/88956315","seeCount":0,"answerUrl":"222"},{"id":721,"newTitle":"对象数组如何去重？（烈熊网络）","subDes":"","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://juejin.im/post/5db556376fb9a0207a6ddce7#heading-7"},{"id":719,"newTitle":"移动端如何设计一个比较友好的Header组件？(携程)","subDes":"","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://juejin.im/post/5db556376fb9a0207a6ddce7#heading-14"},{"id":718,"newTitle":"说出space-between和space-around的区别？（携程）","subDes":"","level":1,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://juejin.im/post/5db556376fb9a0207a6ddce7#heading-15"},{"id":717,"newTitle":"React Hooks当中的useEffect是如何区分生命周期钩子的（喜马拉雅）","subDes":"","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://juejin.im/post/5db556376fb9a0207a6ddce7#heading-20"},{"id":716,"newTitle":"React实现的移动应用中，如果出现卡顿，有哪些可以考虑的优化方案（喜马拉雅）","subDes":"","level":1,"viewCount":0,"newContent":"(1) 增加shouldComponentUpdate钩子对新旧props进行比较，如果值相同则阻止更新，避免不必要的渲染，或者使用PureReactComponent替代Component，其内部已经封装了shouldComponentUpdate的浅比较逻辑；\n\n(2) 对于列表或其他结构相同的节点，为其中的每一项增加唯一key属性，以方便React的diff算法中对该节点的复用，减少节点的创建和删除操作；\n\n(3) render函数中减少类似onClick={() => {doSomething()}}的写法，每次调用render函数时均会创建一个新的函数，即使内容没有发生任何变化，也会导致节点没必要的重渲染，建议将函数保存在组件的成员对象中，这样只会创建一次；\n\n(4) 组件的props如果需要经过一系列运算后才能拿到最终结果，则可以考虑使用reselect库对结果进行缓存，如果props值未发生变化，则结果直接从缓存中拿，避免高昂的运算代价；\n\n(5) webpack-bundle-analyzer分析当前页面的依赖包，是否存在不合理性，如果存在，找到优化点并进行优化。\n\n来源\nhttps://juejin.im/post/5db556376fb9a0207a6ddce7#heading-24","seeCount":0,"answerUrl":"https://juejin.im/post/5db556376fb9a0207a6ddce7#heading-24"},{"id":715,"newTitle":"如何从10000个数中找到最大的10个数(喜马拉雅)","subDes":"","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://juejin.im/post/5db556376fb9a0207a6ddce7#heading-25"},{"id":714,"newTitle":"React实现一个防抖的模糊查询输入框","subDes":"","level":1,"viewCount":0,"newContent":"```\n  // 防抖函数\n  function debounce(fn, wait, immediate) {\n    let timer = null;\n    \n    return function (...args) {\n        let context = this;\n        \n        if (immediate ","seeCount":0,"answerUrl":"https://juejin.im/post/5db556376fb9a0207a6ddce7#heading-27"},{"id":713,"newTitle":"手动封装一个请求函数，可以设置最大请求次数，请求成功则不再请求，请求失败则继续请求直到超过最大次数(流利说)","subDes":"","level":2,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://juejin.im/post/5db556376fb9a0207a6ddce7#heading-28"},{"id":712,"newTitle":"GET和POST的区别（流利说）","subDes":"","level":1,"viewCount":0,"newContent":"(1) GET请求在浏览器回退和刷新时是无害的，而POST请求会告知用户数据会被重新提交；\n\n(2) GET请求可以收藏为书签，POST请求不可以收藏为书签；\n\n(3) GET请求可以被缓存，POST请求不可以被缓存，除非在响应头中包含合适的Cache-Control/Expires字段，但是不建议缓存POST请求，其不满足幂等性，每次调用都会对服务器资源造成影响；\n\n(4) GET请求一般不具有请求体，因此只能进行url编码，而POST请求支持多种编码方式。\n\n(5) GET请求的参数可以被保留在浏览器的历史中，POST请求不会被保留；\n\n(6) GET请求因为是向URL添加数据，不同的浏览器厂商，代理服务器，web服务器都可能会有自己的长度限制，而POST请求无长度限制；\n\n(7) GET请求只允许ASCII字符，POST请求无限制，支持二进制数据；\n\n(8) GET请求的安全性较差，数据被暴露在浏览器的URL中，所以不能用来传递敏感信息，POST请求的安全性较好，数据不会暴露在URL中；\n\n\n(9) GET请求具有幂等性(多次请求不会对资源造成影响)，POST请求不幂等；\n\n(10) GET请求一般不具有请求体，请求中一般不包含100-continue 协议，所以只会发一次请求，而POST请求在发送数据到服务端之前允许双方@[Zz]握手@[Zz]，客户端先发送Expect:100-continue消息，询问服务端是否愿意接收数据，接收到服务端正确的100-continue应答后才会将请求体发送给服务端，服务端再响应200返回数据。\n\nhttps://juejin.im/post/5db556376fb9a0207a6ddce7#heading-30\n来源\n","seeCount":0,"answerUrl":""},{"id":711,"newTitle":"CSS3中transition和animation的属性分别有哪些（哔哩哔哩）","subDes":"","level":1,"viewCount":0,"newContent":"transition 过渡动画：\n(1) transition-property：属性名称\n(2) transition-duration: 间隔时间\n(3) transition-timing-function: 动画曲线\n(4) transition-delay: 延迟\nanimation 关键帧动画：\n(1) animation-name：动画名称\n(2) animation-duration: 间隔时间\n(3) animation-timing-function: 动画曲线\n(4) animation-delay: 延迟\n(5) animation-iteration-count：动画次数\n(6) animation-direction: 方向\n(7) animation-fill-mode: 禁止模式\n\n解析来源\n\nhttps://juejin.im/post/5db556376fb9a0207a6ddce7\n","seeCount":0,"answerUrl":"https://juejin.im/post/5db556376fb9a0207a6ddce7#heading-33"},{"id":710,"newTitle":"forEach，map和filter的区别（哔哩哔哩）","subDes":"","level":1,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://juejin.im/post/5db556376fb9a0207a6ddce7#heading-36"},{"id":709,"newTitle":"跨标签页的通讯方式有哪些（哔哩哔哩）","subDes":"","level":1,"viewCount":0,"newContent":"```\n(1) BroadCast Channel\n(2) Service Worker\n(3) LocalStorage   window.onstorage监听\n(4) Shared Worker   定时器轮询(setInterval)\n(5) IndexedDB   定时器轮询(setInterval)\n(6) cookie   定时器轮询(setInterval)\n(7) window.open   window.postMessage\n(8) Websocket\n\n```\n解析来源\nhttps://juejin.im/post/5db556376fb9a0207a6ddce7","seeCount":0,"answerUrl":""},{"id":708,"newTitle":"实现一个函数判断数据类型(哔哩哔哩)","subDes":"0","level":1,"viewCount":0,"newContent":"```\nfunction getType(obj) {\n   if (obj === null) return String(obj);\n   return typeof obj === 'object' \n   ? Object.prototype.toString.call(obj).replace('[object ', '').replace(']', '').toLowerCase()\n   : typeof obj;\n}\n\n// 调用\ngetType(null); // -> null\ngetType(undefined); // -> undefined\ngetType({}); // -> object\ngetType([]); // -> array\ngetType(123); // -> number\ngetType(true); // -> boolean\ngetType('123'); // -> string\ngetType(/123/); // -> regexp\ngetType(new Date()); // -> date\n```\n\n解析参考\nhttps://juejin.im/post/5db556376fb9a0207a6ddce7#heading-39","seeCount":0,"answerUrl":"https://juejin.im/post/5db556376fb9a0207a6ddce7#heading-38"},{"id":690,"newTitle":"delete 数组的 item，数组的 length 是否会 -1","subDes":"","level":1,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":""},{"id":689,"newTitle":"给出 ['1', '3', '10'].map(parseInt) 执行结果","subDes":"0","level":1,"viewCount":0,"newContent":"0","seeCount":0,"answerUrl":"0"},{"id":688,"newTitle":"http 劫持","subDes":"","level":1,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://juejin.im/post/59ba146c6fb9a00a4636d8b6#heading-2"},{"id":687,"newTitle":"HTTP劫持、DNS劫持与XSS","subDes":"","level":1,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://www.cnblogs.com/coco1s/p/5777260.html"},{"id":686,"newTitle":"xss csrf 的一切","subDes":"","level":1,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://github.com/dwqs/blog/issues/68"},{"id":685,"newTitle":"请说说 CSRF 攻防","subDes":"0","level":1,"viewCount":0,"newContent":"0","seeCount":0,"answerUrl":"https://juejin.im/post/5b6b08956fb9a04fc67c2263 https://juejin.im/post/5bc009996fb9a05d0a055192"},{"id":684,"newTitle":"HTTP劫持、DNS劫持与XSS","subDes":"","level":1,"viewCount":0,"newContent":"","seeCount":0,"answerUrl":"https://www.cnblogs.com/coco1s/p/5777260.html"},{"id":683,"newTitle":"简单说下xss攻击和防御","subDes":"0","level":1,"viewCount":0,"newContent":"0","seeCount":0,"answerUrl":"https://segmentfault.com/a/1190000020402185 https://juejin.im/post/5bad9140e51d450e935c6d64"},{"id":682,"newTitle":"react 有几种事件绑定方式以及区别是什么","subDes":"0","level":1,"viewCount":0,"newContent":"0","seeCount":0,"answerUrl":"https://segmentfault.com/a/1190000011317515"},{"id":681,"newTitle":"redux异步中间件之间的优劣?","subDes":"0","level":2,"viewCount":0,"newContent":"0","seeCount":0,"answerUrl":"https://juejin.im/post/5d5f44dae51d4561df7805b4#heading-16"},{"id":680,"newTitle":"redux中如何进行异步操作?","subDes":"0","level":2,"viewCount":0,"newContent":"0","seeCount":0,"answerUrl":"https://juejin.im/post/5d5f44dae51d4561df7805b4#heading-15"},{"id":679,"newTitle":"redux与mobx的区别?","subDes":"","level":2,"viewCount":0,"newContent":"两者对比:\n\nredux将数据保存在单一的store中，mobx将数据保存在分散的多个store中\nredux使用plain object保存数据，需要手动处理变化后的操作；mobx适用observable保存数据，数据变化后自动处理响应的操作\nredux使用不可变状态，这意味着状态是只读的，不能直接去修改它，而是应该返回一个新的状态，同时使用纯函数；mobx中的状态是可变的，可以直接对其进行修改\nmobx相对来说比较简单，在其中有很多的抽象，mobx更多的使用面向对象的编程思维；redux会比较复杂，因为其中的函数式编程思想掌握起来不是那么容易，同时需要借助一系列的中间件来处理异步和副作用\nmobx中有更多的抽象和封装，调试会比较困难，同时结果也难以预测；而redux提供能够进行时间回溯的开发工具，同时其纯函数以及更少的抽象，让调试变得更加的容易\n\n场景辨析:\n基于以上区别,我们可以简单得分析一下两者的不同使用场景.\nmobx更适合数据不复杂的应用: mobx难以调试,很多状态无法回溯,面对复杂度高的应用时,往往力不从心.\nredux适合有回溯需求的应用: 比如一个画板应用、一个表格应用，很多时候需要撤销、重做等操作，由于redux不可变的特性，天然支持这些操作.\nmobx适合短平快的项目: mobx上手简单,样板代码少,可以很大程度上提高开发效率.\n当然mobx和redux也并不一定是非此即彼的关系,你也可以在项目中用redux作为全局状态管理,用mobx作为组件局部状态管理器来用.\n\n原解析在这儿：\nhttps://juejin.im/post/5d5f44dae51d4561df7805b4#heading-14","seeCount":0,"answerUrl":"https://juejin.cn/post/6844903922453200904#heading-14"},{"id":678,"newTitle":"react-redux是如何工作的?","subDes":"0","level":2,"viewCount":0,"newContent":"0","seeCount":0,"answerUrl":"https://juejin.im/post/5d5f44dae51d4561df7805b4#heading-13"},{"id":677,"newTitle":"redux的工作流程?","subDes":null,"level":2,"viewCount":0,"newContent":null,"seeCount":0,"answerUrl":"https://juejin.im/post/5d5f44dae51d4561df7805b4#heading-12"},{"id":676,"newTitle":"你对 Time Slice的理解?","subDes":"0","level":2,"viewCount":0,"newContent":"0","seeCount":0,"answerUrl":"https://juejin.im/post/5d5f44dae51d4561df7805b4#heading-11"},{"id":674,"newTitle":"mixin、hoc、render props、react-hooks的优劣如何？","subDes":"0","level":2,"viewCount":0,"newContent":"0","seeCount":0,"answerUrl":"https://juejin.im/post/5d5f44dae51d4561df7805b4#heading-9"},{"id":673,"newTitle":"React如何进行组件/逻辑复用?","subDes":"0","level":2,"viewCount":0,"newContent":"0","seeCount":0,"answerUrl":"https://juejin.im/post/5d5f44dae51d4561df7805b4#heading-8"},{"id":672,"newTitle":"React有哪些性能优化的手段?","subDes":"0","level":2,"viewCount":0,"newContent":"0","seeCount":0,"answerUrl":"https://juejin.im/post/5d5f44dae51d4561df7805b4#heading-7"},{"id":671,"newTitle":"React组件通信如何实现?","subDes":null,"level":2,"viewCount":0,"newContent":null,"seeCount":0,"answerUrl":"https://juejin.im/post/5d5f44dae51d4561df7805b4#heading-6"},{"id":670,"newTitle":"setState到底是异步还是同步?","subDes":"0","level":2,"viewCount":0,"newContent":"0","seeCount":0,"answerUrl":"https://juejin.im/post/5d5f44dae51d4561df7805b4#heading-5"},{"id":669,"newTitle":"React的请求应该放在哪个生命周期中?","subDes":"0","level":2,"viewCount":0,"newContent":"0","seeCount":0,"answerUrl":"https://juejin.im/post/5d5f44dae51d4561df7805b4#heading-4"}]}}}}}