高频面试题

高频面试题

[TOC]

一、HTML

1、你都做过哪些兼容性问题?

HTML 兼容性:

h5 新标签只能兼容到 ie9,如果想要兼容 ie 低版本浏览器,需要引入 html5shiv.js 文件,其 cdn 写法如下:

<script src="https://oss.maxcdn.com/libs/html5shiv/3.7.0/html5shiv.js"></script>

CSS 兼容性:

1.1、媒体查询兼容性,ie9 以下不支持媒体查询,需要引入 response.js 文件,其 cdn 写法如下:

<script src="https://oss.maxcdn.com/libs/respond.js/1.3.0/respond.min.js"></script>

1.2、CSS Hack:

1.2.1、属性前缀:例如 IE6 能识别下划线和星号,IE7 能识别星号,但不能识别下划线,IE6~IE10 都认识”\9”,但 firefox 前述三个都不能认识。

.red {
 _color: red; /* ie6 */
 *color: red; /* ie7 */
 color: red\9; /* ie8/9/10 */
}

1.2.2、选择器前缀:例如 IE6 能识别 html .class{},IE7 能识别 +html .class{}或者:first-child+html .class{}。

*.red {} /* ie6 */
+.red {} /* ie7 */

1.2.3、条件注释:

针对所有 IE(注:IE10+已经不再支持条件注释):

<!--[if IE]>IE浏览器显示的内容 <![endif]-->

针对 IE6 及以下版本:

<!--[if lt IE 6]>只在IE6-显示的内容 <![endif]-->

这类 Hack 不仅对 CSS 生效,对写在判断语句里面的所有代码都会生效。

1.3、厂商前缀:谷歌-webkit-、火狐-moz-、IE-ms-、欧朋-o-

1.4、其它兼容性:

1.4.1、ie 老版本浮动造成的双边距问题:display:inline;

1.4.2、图片间隙:父盒子设置 font-size: 0; 或者图片设置 display: block;

1.4.3、块元素默认高度:overflow: hidden;

JavaScript 兼容性:一般使用渐进增强和优雅降级的方式来解决兼容性问题。

// 优雅降级
var xhr = null;
if (XMLHttpRequest) {
  xhr = new XMLHttpRequest();
} else {
  xhr = new ActiveXObject("Microsoft.XMLHTTP");
}

// 渐进增强
// 前边实现上传文件的基本功能
// 后边再判断如果支持拖拽事件,就实现拖拽上传

2、如何提高页面性能?

2.1、图片压缩、合并(精灵图)、使用字体图标代替小图片、使用 base64、图片懒加载

2.2、css、js 的压缩、封装复用

2.3、减少重排操作,例如使用 transform 书写动画效果,在 for 循环结束后再去操作 dom 等

2.3、使用 CDN 网络托管

2.4、数据懒加载、数据按需加载(上拉加载)、分页

2.5、路由懒加载

2.6、利用缓存来缓存文件

2.7、频繁触发的事件进行防抖和节流

2.8、异步加载

2.9、减少闭包,递归优化,使用高效的算法

2.10、webpack 优化:去除无用代码 treeShaking、组件按需加载、使用 chunck、模板预编译等

2.11、字库用 gb2312 不要 utf-8,一个汉字少一个字节


3、谈谈你对 H5 的理解?

Html5 是 Web 中核心语言 HTML 的规范,是 HyperText Markup Language 5 的缩写,H5 提供新的标签元素,使代码变的更有语义;提供了大量 api,如本地存储、离线存储、webworker、websocket、filereader、地理定位、拖拽等;提供了更加酷炫的 CSS3 新特性,如过渡、变形、动画、阴影、渐变等。


4、浏览器从输入网址都看到网页都发生了什么?

4.1、域名解析成 ip 地址

4.2、客户端发送一个带有 SYN 标志的数据包给服务端(三次握手,第一次)

4.3、服务端收到后,回传一个带有 SYN/ACK 标志的数据包以示传达确认信息(三次握手,第二次)

4.4、客户端再回传一个带 ACK 标志的数据包,代表握手结束,连接成功(三次握手,第三次)

4.5、服务端处理数据并返回数据

4.6、客户端请求关闭连接(四次挥手,第一次)

4.7、服务端确认是否还有数据要传输(四次挥手,第二次)

4.8、服务端没有要传输的数据了,准备关闭连接(四次挥手,第三次)

4.9、客户端断开连接(四次挥手,第四次)

4.10、浏览器解析 HTML,生成 DOM 树,解析 CSS,生成 CSS 规则树

4.11、DOM 树和 CSS 规则树合并成渲染树,开始渲染

4.12、执行 JavaScript 脚本


5、重绘和重排?

重排也叫回流,当元素因为规模尺寸,布局,隐藏等改变而需要重新构建时则成为重排。

重绘:一些元素需要更新属性,而这些属性只是影响元素的外观,风格,而不会影响布局则叫重绘。

重绘不一定重排,但是重排一定重绘。


6、缓存?

6.1、浏览器缓存:就是把一个已经请求过的资源拷贝一份存储起来,当下次需要该资源时, 浏览器会根据缓存机制决定直接使用缓存资源还是再次向服务器发送请求。分为:

强制缓存:请求头设置 cache-control:

​ max-age 缓存的时间

​ no-cache:不使用本地缓存。需要使用缓存协商,先与服务器确认返回的响应是否被更改,如果之前的响应中存在 ETag,那么请求的时候会与服务端验证,如果资源未被更改,则可以避免重新下载。

​ no-store:直接禁止浏览器缓存数据,每次用户请求该资源,都会向服务器发送一个请求,每次都会下载完整的资源。

​ public:可以被所有的用户缓存,包括终端用户和 CDN 等中间代理服务器。

​ private:只能被终端用户的浏览器缓存,不允许 CDN 等中继缓存服务器对其缓存。

协商缓存:请求头设置 last-modified/etag

1.Etag 要优于 Last-Modified。Last-Modified 的时间单位是秒,如果某个文件在 1 秒内改变了多次,那么他们的 Last-Modified 其实并没有体现出来修改,但是 Etag 每次都会改变确保了精度;

2.在性能上,Etag 要逊于 Last-Modified,毕竟 Last-Modified 只需要记录时间,而 Etag 需要服务器通过算法来计算出一个 hash 值;

3.在优先级上,服务器校验优先考虑 Etag。

获取资源形式 状态码 发送请求到服务器
强缓存 从缓存取 200(fromcache) 否,直接从缓存取
协商缓存 从缓存取 304(notmodified) 是,通过服务器来告知缓存是否可用

image-20210416160030758

6.2、H5 缓存:

本地存储:localStorage 永久存储、sessionStorage 临时存储

离线缓存:在 html 标签上设置 manifest 属性 引入 cache 文件(CACHE 缓存文件,NETWORK 不缓存文件,FALLBACK 当资源不可访问时,代替的文件)

6.3、更新缓存文件:1、更新 manifest 文件;2、通过 javascript 操作:window.applicationCache.update();3、清除浏览器缓存;4、带版本号,根据版本号判断。


7、状态码?

1 字头:信息,服务器收到请求,需要请求者继续执行操作

2 字头:成功,操作被成功接收并处理

3 字头:重定向,需要进一步的操作以完成请求

4 字头:客户端错误,请求包含语法错误或无法完成请求

5 字头:服务器错误,服务器在处理请求的过程中发生了错误

101:切换协议。

200:请求成功。一般用于 GET 与 POST 请求

203:非授权信息。请求成功。但返回的 meta 信息不在原始的服务器,而是一个副本

204:无内容。服务器成功处理,但未返回内容。

301:永久移动。请求的资源已被永久的移动到新 URI,返回信息会包括新的 URI,浏览器会自动定向到新 URI。

302:临时移动。

304:未修改。所请求的资源未修改,服务器返回此状态码时,不会返回任何资源。

305:使用代理。所请求的资源必须通过代理访问

307:临时重定向。

400:客户端请求的语法错误,服务器无法理解

404:服务器无法根据客户端的请求找到资源(网页)

405:客户端请求中的方法被禁止

500:服务器内部错误,无法完成请求

502:作为网关或者代理工作的服务器尝试执行请求时,从远程服务器接收到了一个无效的响应

503:由于超载或系统维护,服务器暂时的无法处理客户端的请求。

505:服务器不支持请求的 HTTP 协议的版本,无法完成处理


8、!DOCTYPE html 是干什么的,有什么用?

1、声明文档类型是 html5 类型的文档。2、声明了则是标准模式,兼容 ie 高版本;不声明则是混杂模式,兼容 ie 低版本。


9、http1.0 、http1.1 和 http2.0 的区别

1、长链接

HTTP 1.0 需要使用 keep-alive 参数来告知服务器端要建立一个长连接,而 HTTP1.1 默认支持长连接。

HTTP 是基于 TCP/IP 协议的,创建一个 TCP 连接是需要经过三次握手的,有一定的开销,如果每次通讯都要重新建立连接的话,对性能有影响。因此最好能维持一个长连接,可以用个长连接来发多个请求。

2、节约带宽

HTTP 1.1 支持只发送 header 信息(不带任何 body 信息),如果服务器认为客户端有权限请求服务器,则返回 100,否则返回 401。客户端如果接受到 100,才开始把请求 body 发送到服务器。

这样当服务器返回 401 的时候,客户端就可以不用发送请求 body 了,节约了带宽。

3、HOST 域

HTTP1.0 没有 host 域,HTTP1.1 有 host 域。HOST 域就是,web server 上的多个虚拟站点可以共享同一个 ip 和端口

4、多路复用

HTTP2.0 使用了多路复用的技术,做到同一个连接并发处理多个请求,而且并发请求的数量比 HTTP1.1 大了好几个数量级

5、数据压缩

HTTP1.1 不支持 header 数据的压缩,HTTP2.0 使用 HPACK 算法对 header 的数据进行压缩,这样数据体积小了,在网络上传输就会更快。

6、服务器推送

当我们对支持 HTTP2.0 的 web server 请求数据的时候,服务器会顺便把一些客户端需要的资源一起推送到客户端,免得客户端再次创建连接发送请求到服务器端获取。这种方式非常合适加载静态资源。服务器端推送的这些资源其实存在客户端的本地,客户端直接从本地加载这些资源就可以了,不用走网络,速度自然是快很多的。


10、http 和 https 的区别

1、https 协议需要到 ca 申请证书,一般免费证书较少,因而需要一定费用。

2、http 是超文本传输协议,信息是明文传输,https 则是具有安全性的 ssl 加密传输协议。

3、http 和 https 使用的是完全不同的连接方式,用的端口也不一样,前者是 80,后者是 443。

4、http 的连接很简单,是无状态的;HTTPS 协议是由 SSL+HTTP 协议构建的可进行加密传输、身份认证的网络协议,比 http 协议安全。

虽然说 HTTPS 有很大的优势,但其相对来说,还是存在不足之处的:

(1)HTTPS 协议握手阶段比较费时,会使页面的加载时间延长近 50%,增加 10%到 20%的耗电;

(2)HTTPS 连接缓存不如 HTTP 高效,会增加数据开销和功耗,甚至已有的安全措施也会因此而受到影响;

(3)SSL 证书需要钱,功能越强大的证书费用越高,个人网站、小网站没有必要一般不会用。

(4)SSL 证书通常需要绑定 IP,不能在同一 IP 上绑定多个域名,IPv4 资源不可能支撑这个消耗。

(5)HTTPS 协议的加密范围也比较有限,在黑客攻击、拒绝服务攻击、服务器劫持等方面几乎起不到什么作用。最关键的,SSL 证书的信用链体系并不安全,特别是在某些国家可以控制 CA 根证书的情况下,中间人攻击一样可行。


11、TCP 和 UDP 的区别

(1)TCP 是面向连接的,udp 是无连接的即发送数据前不需要先建立链接。

(2)TCP 提供可靠的服务。也就是说,通过 TCP 连接传送的数据,无差错,不丢失,不重复,且按序到达;UDP 尽最大努力交付,即不保证可靠交付。并且因为 tcp 可靠,面向连接,不会丢失数据因此适合大数据量的交换。

(3)TCP 是面向字节流,UDP 面向报文,并且网络出现拥塞不会使得发送速率降低(因此会出现丢包,对实时的应用比如 IP 电话和视频会议等)。

(4)TCP 只能是 1 对 1 的,UDP 支持 1 对 1,1 对多。

(5)TCP 的首部较大为 20 字节,而 UDP 只有 8 字节。

(6)TCP 是面向连接的可靠性传输,而 UDP 是不可靠的。


二、CSS

1、display:none; 和 visibility:hidden;的区别是什么?

display:none; 彻底消失,释放空间。能引发页面的 reflow 回流(重排)。

visibility:hidden; 就是隐藏,但是位置没释放,好比 opacity:0; 不引发页面回流。


2、CSS 优先级和权重值如何计算

内嵌样式>内部样式>外部样式>导入式

!important > 内嵌 1000 >Id 100 > class=[]=伪类 10 > tag=伪元素 1 > ( * + > ~) 0


3、如何触发 BFC,以及 BFC 的作用

BFC:块级格式化上下文 block formatting context,是一个独立渲染区域。规定了内部 box 如何布局,并且与这个区域外部毫不相干。

触发:float 的值不是 none;position 的值不是 static 或者 relative;display 的值是 inline-block、block、table-cell、flex、table-caption 或者 inline-flex;overflow 的值不是 visible。

作用:避免 margin 重叠;自适应两栏布局;清除浮动。


4、CSS 盒模型

盒模型由:外边距 margin、边框 border、内边距 padding、内容 content 四个部分组成

标准盒模型大小=border+padding+content

怪异盒模型大小=content

转怪异盒模型:box-sizing:border-box;

转标准盒模型:box-sizing:content-box;


5、如何水平垂直居中一个元素

5.1、弹性盒子

.box{
  display: flex;
  justify-content: center;
  align-items: center;
}

5.2、定位

.box{
  position: relative;
}
.box .sub{
  position: absolute;
  top: 50%;
  left: 50%;
  transform: translate(-50%, -50%);
  /*margin-left: 负的宽度的一半*/
  /*margin-top: 负的高度的一半*/
}

6、css 实现一个三角形

.triangle{
  width: 0;
  height: 0;
  border: 100px solid transparent;
  border-left-color: red;
}

7、如何实现左边固定宽,右边自适应布局

7.1、圣杯布局

<div id="container">
  <div id="center" class="column"></div>
  <div id="left" class="column"></div>
  <div id="right" class="column"></div>
</div>
#container {
  padding-left: 200px;
  padding-right: 150px;
}
#container .column {
  float: left;
}
#center {
  width: 100%;
}
#left {
  width: 200px;
  margin-left: -100%;
  position: relative;
  right: 200px;
}
#right {
  width: 150px;
  margin-right: -150px;
}

7.2、双飞翼布局

  <div id="container" class="column">
    <div id="center"></div>
  </div>
  <div id="left" class="column"></div>
  <div id="right" class="column"></div>
  #container {
    width: 100%;
  }

.column {
  float: left;
}

#center {
  margin-left: 200px;
  margin-right: 150px;
}

#left {
  width: 200px;
  margin-left: -100%;
}

#right {
  width: 150px;
  margin-left: -150px;
}

7.3、等高布局(假等高)互补的内外边距

.parent{
  overflow: hidden;
}
.left, .right{
  margin-bottom: -10000px;
  padding-bottom: 10000px;
}

7.4、等高布局(真等高)弹性盒子

.parent{
  display: flex;
}
.child{
  flex: 1;
}

7.5、calc

  <div id="left" class="column"></div>
  <div id="center" class="column"></div>
  <div id="right" class="column"></div>
  .column{
    float: left;
  }
  #left{
    width: 100px;
  }
  #right{
    width: 200px;
  }
  #center{
    width: calc(100% - 100px - 200px);
  }

7.6、浮动

  <div id="left" class="column"></div>
  <div id="right" class="column"></div>
  <div id="center"></div>
  #left{
    float: left;
    width: 100px;
  }
  #right{
    float: right;
    width: 200px;
  }
  #center{
    margin-left: 100px;
    margin-right: 200px;
  }

8、如何实现 6px 字体

.font{
  font-size: 12px;
  transform: scale(.5);
}

9、移动端 1px 边框怎么设置

/* 方法1 */
.border{
  width: 100%;
  height: 1px;
  background: red;
}
/* 方法2 */
.border{
  border-image: url(border.png)
}
/* 方法3 */
.border{
  box-shadow: 0 0 0 1px #000;
}

10、px、em、rem、vh、vw 分别是什么

px 物理像素,绝对单位;em 相对于自身字体大小,如果自身没有大小则相对于父级字体大小,如果父级也没有则一层一层向上查找,直到找到 html 为止,相对单位;rem 相对于 html 的字体大小,相对单位;vh 相对于屏幕高度的大小,相对单位;vw 相对于屏幕宽度的大小,相对单位。


11、css 可继承的属性有哪些

可继承的属性:文本类:text-indent、text-align、line-height、word-spacing、letter-spacing、text-transform、direction、color;

字体类:font、font-family、font-weight、font-size、font-style、font-variant、font-stretch、font-size-adjust;

其它类:visibility、caption-side、border-collapse、border-spacing、empty-cells、table-layout、list-style-type、list-style-image、list-style-position、list-style、quotes、cursor、page、page-break-inside、windows、orphans 等


三、JavaScript

1、call、apply、bind 的区别

这三个都是用来定义上下文的,call、apply 会指定上下文并执行函数;而 bind 终身定 死上下文但是不执行函数,并返回新的函数。 其中 call 和 apply 传入参数的形式有别,call 是单独罗列,逗号隔开参数;apply 是数 组。 函数.call(上下文对象,参数,参数,参数); 函数.apply(上下文对象,[参数,参数,参数]);

var obj = {
  a: 10,
};

function fun(b, c) {
  console.log(this.a + b + c);
}

fun.call(obj, 3, 4);
fun.apply(obj, [3, 4]);
fun = fun.bind(obj); // 返回新的函数
fun(3, 4);

2、数据类型有哪些

基本类型:数字 number、字符串 string、布尔 boolean、undefined、null、symbol

引用类型:数组 array、函数 function、对象 object


3、如何检测数据类型

typeof 能够检测:数字、字符串、布尔、undefined、symbol、function

instanceof 能够检测:数组

Object.prototype.toString.call() 万能法


4、各语句的区别

4.1、for 和 for…in 和 for…of 的区别

for 循环,遍历整个数组

for…in 加强循环,不光可以遍历数组,还可以遍历对象和其原型上的方法

for…of 遍历数组和可枚举的对象

4.2、switch 和 if 的区别

switch 用于判断精准的值

if 用于判断值的范围

4.3、while 和 do…while 的区别

while 当符合条件时则执行

do…while 先执行一次,然后再判断是否符合条件,比 while 要多执行一次

4.4、break 和 continue 的区别

break 是跳出当前循环并终止循环

continue 是跳出当前循环并执行下一次循环


5、闭包

闭包就是函数能够记忆住当初定义时候的作用域,不管函数到哪里执行了,永远都能够 记住那个作用域,并且会遮蔽新作用域的变量。可预测状态容器;实现模块化,实现变量的私有封装;可以实现迭代器。 闭包缺点:1.闭包有一个非常严重的问题,那就是内存浪费问题,这个内存浪费不仅仅 因为它常驻内存,更重要的是,对闭包的使用不当的话会造成无效内存的产生;2.性能问题 使用闭包时,会涉及到跨作用域访问,每次访问都会导致性能损失。 因此在脚本中,最好小心使用闭包,它同时会涉及到内存和速度问题。不过我们可以通过把跨作用域变量存储在局部变量中,然后直接访问局部变量,来减轻对执行速度的影响。

function foo() {
  var a = 0;
  function f() {
    a++;
    return a;
  }
  return f;
}
var res = foo();
res();
res();
res();
console.log(res()); // 4 a的值被存储在内存中不会被释放掉

6、原型和原型链

原型:每一个对象类型都有一个隐式原型proto ,每一个函数都有一个显示原型 prototype,该属性指向它的原型对象。

原型链:某个对象的原型又有自己的原型,直到某个对象的原型为 null 为止,组成这条的最后一环,这种一级一级的链就是原型链。


7、继承

7.1、原型链继承

/**
 * 缺点:引用类型的属性被所有实例共享,
 * 在创建Child 的实例时, 不能向Person传参
 */
function Person() {
  this.name = "xiaopao";
}
Person.prototype.getName = function () {
  console.log(this.name);
};
function Child() {}
Child.prototype = new Person();

7.2、借用构造函数继承(经典继承)

/*
优点:
1.避免了引用类型的属性被所有实例共享
2.可以在Child中向Parent传参
缺点:
1.只是子类的实例,不是父类的实例
2.方法都在构造函数中定义,每次创建实例都会创建一遍方法
*/
function Child() {
  Person.call(this);
}

7.3、组合继承

/*
优点:融合原型链继承和构造函数的优点,是JavaScript中最常用的继承模式
缺点:调用了两次父类构造函数
组合继承最大的问题是无论什么情况下,都会调用两次超类型构造函数:一次是在创建子
类型原型的时候,另一次是在子类型构造函数内部)
*/
function Child(name, age) {
  Parent.call(this, name); // 第二次调用 Parent()
  this.age = age;
}
Child.prototype = new Parent(); // 第一次调用 Parent()

7.4、原型式继承

// 缺点: 包含引用类型的属性值始终都会共享相应的值, 这点跟原型链继承一样
function CreateObj(o) {
  function F() {}
  F.prototype = o;
  return new F();
}
var person = {
  name: "xiaopao",
  friend: ["daisy", "kelly"],
};
var person1 = CreateObj(person);

7.5、寄生式继承 可以理解为在原型式继承的基础上增加一些函数或属性

// 缺点:跟借用构造函数一样,每次创建对象都会创建一遍方法

var ob = {
  name: "xiaopao",
  friends: ["lulu", "huahua"],
};

function CreateObj(original) {
  var clone = Object.create(original); //通过调用函数创建一个新对象
  clone.sayHi = function () {
    //以某种方式来增强这个对象
    alert("Hi");
  };
  return clone; //返回这个对象
}

// 上面CreateObj函数 在ECMAScript5 有了一新的规范写法,Object.create(ob) 效果是一样的 , 看下面代码
var ob1 = CreateObj(ob);
console.log(ob1.name); // xiaopao

7.6、寄生组合式继承

// 优点:完美继承
// 缺点:代码繁多,使用起来十分麻烦
function Parent(name) {
  this.name = name;
}

Parent.prototype.sayName = function () {
  console.log(this.name);
};

function Child(name) {
  Parent.call(this, name);
}

function CreateObj(o) {
  function F() {}
  F.prototype = o;
  return new F();
}

function prototype(child, parent) {
  var prototype = CreateObj(parent.prototype);
  prototype.constructor = child;
  child.prototype = prototype;
}
prototype(Child, Parent);

var child = new Child("大圣");
child.sayName();
console.log(child);

7.7、es6 继承

class Child extends Parent {}

8、递归和递归优化

递归就是函数自己调用自己。但是又不能无限的调用自己,需要有一个出口,否则会成为死循环。函数调用是通过栈(stack)这种数据结构实现的,每当进入一个函数调用,栈就会加一层栈帧,每当函数返回,栈就会减一层栈帧。由于栈的大小不是无限的,所以,递归调用的次数过多,会导致栈溢出。

// 循环求1-5的所有数的和
var sum = 0;
for (var i = 1; i <= 5; i++) {
  sum += i;
}
console.log(sum); // 15

//递归实现1-5的所有数的和
function sum(n) {
  if (n === 1) {
    return 1;
  }
  return n + sum(n - 1);
}
console.log(sum(5)); //15

尾递归优化是解决递归调用栈溢出的方法。尾递归是指,在函数返回的时候,调用自身本身,并且,return 语句不能包含表达式。这样,编译器或者解释器就可以把尾递归做优化,使递归本身无论调用多少次,都只占用一个栈帧,不会出现栈溢出的情况。

// 上例递归进行尾递归优化
function sum(n, m = 0) {
  if (n === 1) {
    return 1 + m;
  }
  return sum(n - 1, n + m);
}
console.log(sum(5)); //15

// 或者while优化
function sum(n, m = 0) {
  while (n >= 1) {
    return sum(n - 1, n + m);
  }
  return m;
}
console.log(sum(5)); // 15

9、ajax 工作原理和封装

1.创建 XMLHttpRequest 对象。 2.设置请求方式。open() 3.调用回调函数。onreadystatechange 4.发送请求。send()

function ajax(options) {
  const { type, dataType, data, timeout, url, success, error } = options;
  var params = formatParams(data);
  var xhr;
  //考虑兼容性
  if (window.XMLHttpRequest) {
    xhr = new XMLHttpRequest();
  } else if (window.ActiveObject) {
    //兼容IE6以下版本
    xhr = new ActiveXobject("Microsoft.XMLHTTP");
  }
  //启动并发送一个请求
  if (type == "GET") {
    xhr.open("GET", url + "?" + params, true);
    xhr.send();
  } else if (type == "POST") {
    xhr.open("post", url, true);
    //设置表单提交时的内容类型
    //Content‐type数据请求的格式
    xhr.setRequestHeader("Content‐type", "application/x‐www‐form‐urlencoded");
    xhr.send(params);
  }

  // 设置有效时间
  setTimeout(function () {
    if (xhr.readySate != 4) {
      xhr.abort();
    }
  }, timeout);

  // 接收
  // options.success成功之后的回调函数 options.error失败后的回调函数
  //xhr.responseText,xhr.responseXML 获得字符串形式的响应数据或者XML形式的响应数据
  xhr.onreadystatechange = function () {
    if (xhr.readyState == 4) {
      var status = xhr.status;
      if ((status >= 200 && status < 300) || status == 304) {
        success && success(xhr.responseText, xhr.responseXML);
      } else {
        error && error(status);
      }
    }
  };
}

//格式化请求参数
function formatParams(data) {
  var arr = [];
  for (var name in data) {
    arr.push(encodeURIComponent(name) + "=" + encodeURIComponent(data[name]));
  }
  arr.push(("v=" + Math.random()).replace(".", ""));
  return arr.join("&");
}

10、跨域

跨域是指一个域下的文档或脚本试图去请求另一个域下的资源,这里跨域是广义的。其 实我们通常所说的跨域是狭义的,是由浏览器同源策略限制的一类请求场景。同源策略 SOP(Same origin policy)是一种约定,由 Netscape 公司 1995 年引入浏览器,它是浏览器最核心也最基本的安全功能,如果缺少了同源策略,浏览器很容易受到 XSS、CSFR 等攻击。所谓同源是指”协议+域名+端口”三者相同,即便两个不同的域名指向同一个 ip 地 址,也非同源。

方法 1:跨域资源共享 CORS 跨域,就是服务端在 HTTP 返回头上加上“AccessControll-Allow-Origin:*”。 “Access-Controll-Allow-METHODS:GET, POST” DELETE、PATCH 请求类型会发出 OPTIONS 预检请求。

方法 2:代理跨域,webpack-dev-server 里面的 proxy 配置项。config 中的 ProxyTable

方法 3:JSONP,利用页面 srcipt 没有跨域限制的漏洞,用 script 的 src 引入它,然后页 面内定义回调函数,jQuery 中$.ajax({dataType: ‘jsonp’})。

方法 4: iframe 跨域,配合 window.name 或者 location.hash 或者 document.domain 一起使用

方法 5:nginx 反向代理接口跨域,通过 nginx 配置一个代理服务器(域名与 domain1 相同,端口不同)做跳板机,反向代理访问 domain2 接口,并且可以顺便修改 cookie 中 domain 信息,方便当前域 cookie 写入,实现跨域登录。

方法 6:jquery 的 ajax 跨域,dataType:’jsonp’


11、事件流和事件委托

事件流一般分三个阶段:1、捕获阶段(由外向内) 2、目标阶段 (执行阶段) 3、冒泡阶段(由内向外)

阻止事件冒泡 e.stopPropagation() 阻止默认动作 e.preventDefault()

事件委托:就是把事件委托给父级,利用事件冒泡,只指定一个事件处理程序,就可以管理某一类型的所有事件。


12、事件循环

image-20210415153418143

同步任务进入主线程,异步任务进入 Event Table 并注册函数 当指定的事情完成时,Event Table 会将这个函数移入 Event Queue。 主线程内的任务执行完毕为空,会去 Event Queue 读取对应的函数,进入主线程执 行。 上述过程会不断重复,也就是常说的 Event Loop(事件循环)。

image-20210415153548623

console.log("script start");

setTimeout(function () {
  console.log("setTimeout");
}, 0);

Promise.resolve()
  .then(function () {
    console.log("promise1");
  })
  .then(function () {
    console.log("promise2");
  });

console.log("script end");
/* 执行结果为:script start, script end, promise1, promise2, setTimeout因为Promise是微任务,主线程会在同步任务做完后先清空微任务队列,再执行宏任务队列 */

微任务是由 JavaScript 自身发起,包括:process.nextTick、promise、MutationObserver

宏任务是由宿主发起的,如浏览器、node。包括:setTimeout、setInterval、setImmediate、postMessage


13、防抖和节流

// 节流:在计时器内部清除计时器,有节奏的执行事件
function throttle(callback, delay = 1000) {
  let timer = null;
  function f() {
    if (!timer) {
      timer = setTimeout(() => {
        callback && callback.call(this);
        clearTimeout(timer);
        timer = null;
      }, delay);
    }
  }
  return f;
}

// 防抖:在计时器前边清除计时器,只执行最后一次事件,能够无限延长执行时间
function debounce(callback, delay = 1000) {
  let timer = null;
  function f() {
    clearTimeout(timer);
    timer = setTimeout(() => {
      callback && callback.call(this);
    }, delay);
  }
  return f;
}

14、深克隆和浅克隆

浅克隆:同值也同址。浅拷贝是指源对象与拷贝对象共用一份实体,仅仅是引用的变量不同(名称不同)。对其中任何一个对象的改动都会影响另外一个对象。如:Object.assign;=等号赋值;slice 截取。

深克隆:同值不同址。深拷贝是指源对象与拷贝对象互相独立,其中任何一个对象的改动都不会对另外一个对象造成影响。如:JSON.parse(JSON.stringify());

function deepClone(target) {
  // 定义一个变量
  let result;
  // 如果当前需要深拷贝的是一个对象的话
  if (typeof target === "object") {
    // 如果是一个数组的话
    if (Array.isArray(target)) {
      result = []; // 将result赋值为一个数组,并且执行遍历
      for (let i in target) {
        // 递归克隆数组中的每一项
        result.push(deepClone(target[i]));
      }
      // 判断如果当前的值是null的话;直接赋值为null
    } else if (target === null) {
      result = null;
      // 判断如果当前的值是一个RegExp对象的话,直接赋值
    } else if (target.constructor === RegExp) {
      result = target;
    } else {
      // 否则是普通对象,直接for in循环,递归赋值对象的所有值
      result = {};
      for (let i in target) {
        result[i] = deepClone(target[i]);
      }
    }
    // 如果不是对象的话,就是基本数据类型,那么直接赋值
  } else {
    result = target;
  }
  // 返回最终结果
  return result;
}

15、cookie、sessionStorage 和 localStorage 的区别

15.1、cookie 数据始终在同源的 http 请求中携带(即使不需要),即 cookie 在浏览器和服务器间来回传递,而 sessionStorage 和 localStorage 不会自动把数据发送给服务器,仅在本地保存。cookie 数据还有路径(path)的概念,可以限制 cookie 只属于某个路径下
15.2、存储大小限制也不同,cookie 数据不能超过 4K,同时因为每次 http 请求都会携带 cookie、所以 cookie 只适合保存很小的数据,如会话标识。sessionStorage 和 localStorage 虽然也有存储大小的限制,但比 cookie 大得多,可以达到 5M 或更大
15.3、数据有效期不同,sessionStorage:仅在当前浏览器窗口关闭之前有效;localStorage:始终有效,窗口或浏览器关闭也一直保存,因此用作持久数据;cookie:只在设置的 cookie 过期时间之前有效,即使窗口关闭或浏览器关闭
15.4、作用域不同,sessionStorage 不在不同的浏览器窗口中共享,即使是同一个页面;localstorage 在所有同源窗口中都是共享的;cookie 也是在所有同源窗口中都是共享的
15.5、web Storage 支持事件通知机制,可以将数据更新的通知发送给监听者
15.6、web Storage 的 api 接口使用更方便


16、get 和 post 请求的区别

GET 在浏览器回退时是无害的,而 POST 会再次提交请求。

GET 产生的 URL 地址可以被 Bookmark,而 POST 不可以。

GET 请求会被浏览器主动 cache,而 POST 不会,除非手动设置。

GET 请求只能进行 url 编码,而 POST 支持多种编码方式。

GET 请求参数会被完整保留在浏览器历史记录里,而 POST 中的参数不会被保留。

GET 请求在 URL 中传送的参数是有长度限制的,而 POST 么有。

对参数的数据类型,GET 只接受 ASCII 字符,而 POST 没有限制。

GET 比 POST 更不安全,因为参数直接暴露在 URL 上,所以不能用来传递敏感信息。

GET 参数通过 URL 传递,POST 放在 Request body 中


17、new 操作符都做了哪些事情

构造函数中没有显示的创建 Object 对象,实际上后台自动创建了一个空对象,直接给 this 对象赋值属性和方法,this 即指向创建的对象。没有 return 返回值,后台自动返回了该对象,该对象继承构造函数的原型

// 模拟构造函数实现
var Book = function (name) {
  this.name = name;
};
//正常用法
var js = new Book("js");
//使用代码模拟,在非IE浏览器中测试,IE浏览器不支持
var javascript = {};
javascript.__proto__ = Book.prototype;
Book.call(javascript, "js");

18、XSS 攻击和 CSRF 攻击

XSS:跨站脚本攻击 Cross site script,因叫 css 容易让人误会所以改成了 xss。比如一个 JSON 数据:

var obj = [
  {
    id: 1,
    name: "<script>alert('哈哈哈')</script>",
    age: 12,
  },
];

在不该出现 script 代码的地方出现了,引发一些潜在的危险。 XSS 漏洞,能让人们在网页里面插入一段有功能的语句。 XSS 全称“跨站脚本”,是注入攻击的一种。其特点是不对服务器端造成任何伤害, 而是通过一些正常的站内交互途径,例如发布评论,提交含有 JavaScript 的内容文本。这时服务器端如果没有过滤或转义掉这些脚本,作为内容发布到了页面上,其他用户访问这个 页面的时候就会运行这些脚本。 防范: ① 用正则表达式阻止用户提交带有<、eval、script 等危险字眼的语句 ② 显示的时候不要直接用 innerHTML,而是用 innerText,或者将<转义。

CSRF 的全称是“跨站请求伪造”,而 XSS 的全称是“跨站脚本”。看起来有点相 似,它们都是属于跨站攻击——不攻击服务器端而攻击正常访问网站的用户,但前面说 了,它们的攻击类型是不同维度上的分类。CSRF 顾名思义,是伪造请求,冒充用户在站内 的正常操作。我们知道,绝大多数网站是通过 cookie 等方式辨识用户身份(包括使用服务 器端 Session 的网站,因为 Session ID 也是大多保存在 cookie 里面的),再予以授权的。 所以要伪造用户的正常操作,最好的方法是通过 XSS 或链接欺骗等途径,让用户在本机(即 拥有身份 cookie 的浏览器端)发起用户所不知道的请求。 就是说,如果用户不老老实实写姓名,写了一个个<script>叫做 XSS。如果进一步的,写了一个$.post()发了 document.cookie 就是 CSRF 了。解决方法: ① 用 token 验证,验证用户的 IP 地址生成 MD5 码,更安全的验证方法 ② 防住 XSS。


19、垃圾回收机制

一般来说没有被引用的对象就是垃圾,就是要被清除, 有个例外如果几个对象引用形成一个环,互相引用,但根本访问不到它们,这几个对象也是垃圾,也要被清除。 垃圾回收的方法主要有两种:一种是标记清除,即用完之后的变量 赋值成 null,另一种 是引用计数,将使用完的对象的引用计数,如果为 0 则回收


20、常用 DOM 操作

createElement 创建

appendChild 末尾添加

insertBefore 前边插入

cloneNode(true) 克隆

removeChild() 移除

parentNode 父节点

childNodes // 全部子节点

firstChild // 第一个子节点

lastChild // 最后一个子节点

previousElementSibling// 上一个兄弟节点

nextElementSibling// 下一个兄弟节点

获取 dom 节点:document.getElementById() 、document.getElementsByTagName() 、document.getElementsByClassName() 、document.getElementsByName() 、document.querySelector() 、document.querySelectorAll()


21、AMD、CMD、ES6、CommonJS 的区别

CommonJS:模块引用(require) 模块输出(exports) 模块标识(module) ES6:模块引用(import) 模块输出(export) 前者支持动态导入,也就是 require(${path}/xx.js),后者目前不支持。 前者是同步导入,因为用于服务端,文件都在本地,同步导入即使卡住主线 程影响也不大。而后者是异步导入,因为用于浏览器,需要下载文件,如果也采用同 步导入会对渲染有很大影响。 前者在导出时都是值拷贝,就算导出的值变了,导入的值也不会改变,所以 如果想更新值,必须重新导入一次。但是后者采用实时绑定的方式,导入导出的值都 指向同一个内存地址,所以导入值会跟随导出值变化 AMD、CMD 都使用 define 定义模块,require 引入模块,区别在于 AMD 是前置依赖, CMD 是就近依赖

// AMD   依赖必须一开始就声明
define(["./a", "./b"], function (require, factory) {
  // do something...
});

// CMD
define(function (require, factory) {
  var a = require("./a"); // 依赖就近书写
  // do something...
});

四、ES6

1、let、const、var 的区别

var 声明变量存在变量提升,let 和 const 不存在变量提升

let、const 都是块级局部变量 ,存在暂时性死区

const 的特性和 let 完全一样,不同的只是 const 声明时候必须赋值,赋值基本类型时,只能进行一次赋值,即声明后不能再修改,赋值引用类型时,内存地址不能修改

同一作用域下 let 和 const 不能声明同名变量,而 var 可以


2、箭头函数和普通函数的区别

2.1、箭头函数的 this 是定义时决定的,普通函数是看调用方法。

2.2、箭头函数不能成为构造函数

2.3、箭头函数不能使用 async/await

2.4、箭头函数不能使用 Generator 函数,不能使用 yeild 关键字

2.5、箭头函数不能使用 call、apply、bind 来修改 this 指向

2.6、箭头函数不绑定 arguments

2.7、箭头函数不具有 prototype 原型对象,不具有 super


3、promise 的实现原理和封装

promise 一共有三种状态,分别是 pedding 初始状态 、resolved 成功的状态、 rejected 失败的状态。传入两个参数,一个是 resolve,执行 then 的方法,一个是 reject,执行 catch 的方法或者 then 的第二个参数的回调。promise 一旦状态改变就不可在修改。promise 的链式调用实际上是返回的一个新的 promise,而非 return this。

// 简版promise
function Promise(executor) {
  //executor执行器
  let self = this;
  self.status = "pending"; //等待态
  self.value = undefined; // 表示当前成功的值
  self.reason = undefined; // 表示是失败的值
  function resolve(value) {
    // 成功的方法
    if (self.status === "pending") {
      self.status = "resolved";
      self.value = value;
    }
  }
  function reject(reason) {
    //失败的方法
    if (self.status === "pending") {
      self.status = "rejected";
      self.reason = reason;
    }
  }
  executor(resolve, reject);
}

Promise.prototype.then = function (onFufiled, onRejected) {
  let self = this;
  if (self.status === "resolved") {
    onFufiled(self.value);
  }
  if (self.status === "rejected") {
    onRejected(self.reason);
  }
};

4、forEach、for in、for of 三者区别

forEach 更多的用来遍历数组
for in 一般用来遍历对象或 json,可以遍历对象的原型
for of 一遍用来遍历数组对象和可枚举对象,不能遍历原型
for in 循环出的是 key,for of 循环出的是 value


5、set、map 分别是什么

set 对象:允许你存储任何类型的唯一值,无论是原始值或者是对象引用。Set 是值得集合,不能通过 get 方法获取值,因为 set 只有值。能通过迭代器进行 for…of 遍历。Set 的值是唯一的可以做数组去重。

var set = new Set(["1", undefined, {}, 2, [3, 4]]);
set.size; // 5
set.add(5);
set.delete("1"); // true
set.has("2"); // false
set.keys(); // SetIterator
set.values(); // SetIterator
set.entries(); // SetIterator
// set对象可遍历
set.forEach();

map 对象:Map 对象保存键值对。任何值(对象或者原始值) 都可以作为一个键或一个值。

var myMap = new Map();
var keyObj = {};
myMap.set(keyObj, "和键 keyObj 关联的值");
myMap.get(keyObj); // "和键 keyObj 关联的值"
myMap.size; // 1
myMap.has(keyObj); // true
myMap.delete(keyObj); // true
myMap.clear();
myMap.keys(); // MapIterator 键组成的对象
myMap.values(); // MapIterator  值组成的对象
myMap.entries(); // MapIterator 键值对组成的对象
// map对象可遍历
myMap.forEach((value, key) => {
  console.log(value, key);
});
for (var [key, value] of myMap) {
  console.log(key, value); //  {}, "和键 keyObj 关联的值"
}

6、symbol 的理解

ES6 引入了一种新的原始数据类型 Symbol ,表示独一无二的值,最大的用法是用来定义对象的唯一属性名

let sy = Symbol("KK");
console.log(sy);   // Symbol(KK)
typeof(sy);        // "symbol"

// 相同参数 Symbol() 返回的值不相等
let sy1 = Symbol("kk");
sy === sy1;       // false

// 写法1
let syObject = {};
syObject[sy] = "kk";
console.log(syObject);    // {Symbol(key1): "kk"}

// 写法2
let syObject = {
  [sy]: "kk"
};
console.log(syObject);    // {Symbol(key1): "kk"}

// 写法3
let syObject = {};
Object.defineProperty(syObject, sy, {value: "kk"});
console.log(syObject);   // {Symbol(key1): "kk"}

Symbol 值作为属性名时,该属性是公有属性不是私有属性,可以在类的外部访问。但是不会出现在 for...infor...of 的循环中,也不会被 Object.keys() 、 Object.getOwnPropertyNames() 返回。如果要读取到一个对象的 Symbol 属性,可以通过 Object.getOwnPropertySymbols() 和 Reflect.ownKeys() 取到。

for (let i in syObject) {
  console.log(i);
} // 无输出

Object.keys(syObject); // []
Object.getOwnPropertySymbols(syObject); // [Symbol(key1)]
Reflect.ownKeys(syObject); // [Symbol(key1)]

7、新增哪些数组方法

forEach:forEach()会遍历数组, 循环体内没有返回值,forEach()循环不会改变原来数组的内容, forEach()有三个参数, 第一个参数是当前元素, 第二个参数是当前元素的索引, 第三个参数是当前元素所属的数组

map:map()的主要作用, 其实是创建一个新的数组

filter:filter()主要是过滤的, 用来过滤数组中不满足条件的元素, 把满足条件的元素放到新的数组里, 并且不会改变原数组

every:会遍历数组, 在循环体内写条件, 如果每一项都是 true, 就会返回 true, 只要有一个是 false, 就会返回 false

some:遍历数组的每一项, 然后根据循环体内的条件去判断, 只要有一个是 true, 就会停止循环

reduce:接收一个函数作为累加器, 数组中每个值(从左到右)开始缩减, 最终为一个值

Array.from():用于将两类对象变成数组。一类是类数组对象,一类是可遍历对象。

Array.of():用于将一组值,转换为数组

find() 和 findIndex():用于找出第一个符合条件的数组成员和下标。

fill():方法使用给定值填充一个数组

includes():检查是否包含某个值


8、新增哪些字符串方法

startsWith():返回布尔值,表示参数字符串是否在原字符串的头部

endsWith():返回布尔值,表示参数字符串是否在原字符串的尾部

includes():返回布尔值,表示是否找到了参数字符串

repeat():返回一个新字符串,表示将原字符串重复 n 次

padStart():开始位置填充

padEnd():结束位置填充

trimStart():消除字符串头部的空格

trimEnd():消除字符串尾部的空格

matchAll():返回一个正则表达式在当前字符串的所有匹配


9、新增哪些对象方法

Object.is():判断两个值是否相等

Object.assign():用于将所有可枚举属性的值从一个或多个源对象复制到目标对象。它将返回目标对象。

Object.getOwnPropertyDescriptors():返回指定对象所有自身属性(非继承属性)的描述对象

Object.setPrototypeOf(),Object.getPrototypeOf():用来读取或设置当前对象的 prototype 对象

Object.keys():用于返回一个数组,成员的参数是对象自身的所有可遍历属性的键名

Object.values():返回值是一个数组,成员是对象自身的(不含继承的)所有可遍历属性的值。

Object.entries():返回一个数组,成员是对象自身(不含继承的)所有可遍历属性的键值对数组,Symbol 属性的值会被过滤。

Object.fromEntries():是 Object.entries 的逆操作,将一个键值对数组转为对象。


10、async…await

async…await 是 Generator 函数的语法糖,将*改成 async,将 yield 换成 await。 是对 Generator 函数的改进, 返回 promise。 异步写法同步化,遇到 await 先返回,执行完异步再执行接下来的. 内置执行器, 无需 next()


五、Vue

1、生命周期都有哪些,以及在这些生命周期中都做过哪些事情

beforeCreate 创建之前;无法获取响应数据

created 创建之后,可以在这加个 loading 事件和进行数据请求

beforeMount 挂载前 ,在这结束 loading,还做一些初始数据的获取,实现函数自执行

mounted 挂载后 ,在这发起后端请求,拿回数据,配合路由钩子做一些事情

beforeUpdate 数据更新之前

updated 数据更新完成之后

beforeDestroy 销毁之前 ,你确认删除 XX 吗?或者确认退出吗?

destroyed 销毁之后 ,当前组件已被删除,清空相关内容,在这获取不到 dom 了


2、组件通信

父传子:props、$attrs/$listeners、$children、$root、provide/inject、$refs

子传父:$emit、$parent、

同级传:eventBus、vuex


3、页面通信

url 拼接参数:”/a?a1=a1”,接收页面:this.$route.query.a1

query 传参:{path: ‘a’, query: {a2:’a2’}},接收页面:this.$route.query.a2

params 传参:{name: ‘a’, params: {a3:’a3’}},接收页面:this.$route.params.a3

动态路由传参:/path/a4 ,接收页面:this.$route.params.id,路由:path: “/a/:id”


4、$set 是干什么的

当数据变化但没有更新视图时使用,例如对象新增加的属性,数组新增加的成员

this.$set(obj, "key", "value");

5、$nextTick 是干什么的

在下次 DOM 更新循环结束之后执行延迟回调。在修改数据之后立即使用这个方法,获取更新后的 DOM。

例如:在 created 生命周期中想要操作 dom 就可以使用

this.$nextTick(()=>{ … })可以在 mounted 之前的生命周期中操作 dom


6、mixin 是干什么的

提供了一种非常灵活的方式,来分发 Vue 组件中的可复用功能。一个混入对象可以包含任意组件选项。当组件使用混入对象时,所有混入对象的选项将被“混合”进入该组件本身的选项。

var mixin = {
  data: function () {
    return {
      message: "hello",
      foo: "abc",
    };
  },
};

new Vue({
  mixins: [mixin],
  data: function () {
    return {
      message: "goodbye",
      bar: "def",
    };
  },
  created: function () {
    console.log(this.$data);
    // => { message: "goodbye", foo: "abc", bar: "def" }
  },
});

7、简单说说 MVVM 的理解

MVVM 是 Model-View-ViewModel 的缩写。 Model 代表数据模型,也可以在 Model 中定义数据修改和操作的业务逻辑。 View 代表 UI 组件,它负责将数据模型转化成 UI 展现出来。 ViewModel 监听模型数据的改变和控制视图行为、处理用户交互,简单理 解就是一个同步 View 和 Model 的对象,连接 Model 和 View。 在 MVVM 架构下,View 和 Model 之间并没有直接的联系,而是通过 ViewModel 进行交互,Model 和 ViewModel 之间的交互是双向的, 因此 View 数据的变化会同步到 Model 中,而 Model 数据的变化也 会立即反应到 View 上。 ViewModel 通过双向数据绑定把 View 层和 Model 层连接了起 来,而 View 和 Model 之间的同步工作完全是自动的,无需人为干涉,因此开发者只需关 注业务逻辑,不需要手动操作 DOM, 不需要关注数据状态的同步问题,复杂的数据状态维 护完全由 MVVM 来统一管理。


8、watch 和 computed 的区别

Watch 只能监听 data 中的数据变化,computed 不需要,watch 可以进行异步操作, computed 不可以,computed 不修改原始数据,通过 return 返回处理的数据,可以包含大 量的逻辑运算


9、v-if 和 v-show 的区别

9.1、v-show 只是简单的控制元素的 display 属性,而 v-if 才是条件渲染(条件为真,元 素将会被渲染,条件为假,元素会被销毁);

9.2、v-show 有更高的首次渲染开销,而 v-if 的首次渲染开销要小的多;

9.3、v-if 有更高的切换开销,v-show 切换开销小;

9.4、v-if 有配套的 v-else-if 和 v-else,而 v-show 没有

9.5、v-if 可以搭配 template 使用,而 v-show 不行


10、为什么不能 v-for 和 v-if 一起使用

v-for 优先级是比 v-if 高

永远不要把 v-if 和 v-for 同时用在一个元素上,带来性能方面的浪费(每次渲染都会先循环再进行条件判断)

如果避免出现这种情况,则在外层嵌套 template (页面渲染不生成 dom 节点),再这一层进行 v-if 判断,然后再内部进行 v-for 循环


<template v-if="isShow">
 <p v-for="item in items">
</template>

如果条件出现再循环内部,可通过计算属性 computed 提前过滤掉那些不需要显示的项

computed:{
 items:function(){
  return this.list.filter(function(item){
   return item.isShow
  })
 }
}

11、key 的作用是什么,值写 index 和 id 哪个更好

key 是为每个 vnode 指定唯一的 id,在同级 vnode 的 Diff 过程中,可以根据 key 快速的进 行对比,来判断是否为相同节点, 利用 key 的唯一性生成 map 对象来获取对应节点,比遍历方式更快,指定 key 后,可以 保证渲染的准确性(尽可能的复用 DOM 元素。)赋值时应优先使用 id。


12、过滤器怎么使用

// 全局使用
Vue.filter('globalFilter', function(){
    // ...
})

// 局部使用
filters: {
    formatMoney(num) {
  // ...
    },
}
<p>过滤器{{ money | formatMoney }}</p>

13、vuex 五大核心分别是干什么的

state:Vuex 中的基本数据,辅助函数 mapState

getters:即从 store 的 state 中派生出的状态,有点类似计算属性,辅助函数 mapGetters

mutations:是更改 Vuex 中的 store 中的状态的唯一方法,是同步的,辅助函数 mapMutations

actions:Action 提交的是 mutation,而不是直接变更状态。 Action 可以包含任意异步操作。辅助函数 mapActions

Modules:Vuex 允许我们将 store 分割到模块(module)。每个模块拥有自己的 state、mutations、actions、getters、甚至是嵌套子模块——从上至下进行类似的分割


14、如何调用 mutations 和 actions 的方法

调用mutations:$store.commit('mutations中定义的方法')

调用actions:$store.dispatch('actions中定义的方法')

actions调用mutations中的方法:

fn(context){

    context.commit('mutations中定义的方法');

}

15、vue-router 常写属性都有什么

router-link 常用属性:

to 表示目标路由的链接

replace 设置 replace 属性的话,当点击时,会调用 roter.replace()而不是 router.push(),所以导航后不会留下 history 记录,也就是不能回退到上一个页面

append 设置 append 属性后,则在当前路径前添加基路径,例如,我们从/a 导航到一个相对路径 b,如果没有配置 append,则路径为/b,如果配了,则为/a/b

tag 有时候想要<router-link>渲染成某种标签,例如<li>。于是我们使用 tag prop 类指定何种标签,同样它还是会监听点击,触发导航。

active-class 设置链接激活时使用的 css 类名。默认值可以通过路由的构造选项 linkActiveClass 来全局配置, 默认值为 ‘router-link-active‘

exact “是否激活”,默认是 false 。

vue-router 常用属性:

path 路由路径

name 路由名字

component 导入路由组件

redirect 路由重定向

mode 路由模式

children 子路由

meta 路由元信息


16、路由守卫都有哪些以及都做过哪些事情,三个参数分别是干什么的

全局守卫:beforeEach(登录拦截)、afterEach

路由独享守卫:beforeEnter(部分路由的登录拦截)

组件内守卫:beforeRouteEnter(权限管理)、beforeRouteUpdate、beforeRouteLeave

路由全局解析守卫:beforeResolve(这里根据单页面 name 的指向不同,去访问的接口域名也不同)

三个参数:to:去哪,from:从哪来,next:下一步

当从a页面离开进入b页面时触发的生命周期
    1.beforeRouteLeave:路由组件的组件离开路由前钩子,可取消路由离开。
    2.beforeEach: 路由全局前置守卫,可用于登录验证、全局路由loading等。
    3.beforeEnter: 路由独享守卫
    4.beforeRouteEnter: 路由的组件进入路由前钩子。
    5.beforeResolve:路由全局解析守卫
    6.afterEach:路由全局后置钩子
    7.beforeCreate:组件生命周期,不能访问this8.created:组件生命周期,可以访问this,不能访问dom。
    9.beforeMount:组件生命周期
    10.deactivated: 离开缓存组件a,或者触发a的beforeDestroy和destroyed组件销毁钩子。
    11.mounted:访问/操作dom。
    12.activated:进入缓存组件,进入a的嵌套子组件(如果有的话)13.执行beforeRouteEnter回调函数next。

17、hash 和 history 模式的区别

hash 模式就是 url 后面写#锚点,由于 hash 值变化不会导致浏览器向服务器发出请求,而且 hash 改变会触发 hashchange 事件(hashchange 只能改变 # 后面的 url 片段);更关键的一点是,因为 hash 发生变化的 url 都会被浏览器记录下来,从而你会发现浏览器的前进后退都可以用了,所以人们在 html5 的 history 出现前,基本都是使用 hash 来实现前端路由的

history 模式:hash 能兼容到 IE8, history 只能兼容到 IE10;hash 本来是拿来做页面定位的,如果拿来做路由的话,原来的锚点功能就不能用了。其次,hash 的传参是基于 url 的,如果要传递复杂的数据,会有体积的限制,而 history 模式不仅可以在 url 里放参数,还可以将数据存放在一个特定的对象中。history api 可以分为两大部分:切换(back、forward、go)和修改(pushState,replaceState)。history 模式的问题:就怕刷新,


18、说说常用指令有哪些,如何自定义指令

v-if :如果是真则渲染节点,否则不渲染节点

v-if、v-else 和 v-else-if :类似 js 的 if…else 判断语句

v-show :通过 display:none;控制元素显示隐藏

v-for :循环,v-for 的优先级高于 v-if,不推荐一起使用

v-bind :绑定属性,

v-on :绑定事件,

.stop 阻止事件继续传播

.prevent 事件不再重载页面

.capture 使用事件捕获模式,即元素自身触发的事件先在此处处理,然后才交由内部元素进行处理

.self 只当在 event.target 是当前元素自身时触发处理函数

.once 事件将只会触发一次

.passive 告诉浏览器你不想阻止事件的默认行为

v-model :数据双向绑定

.lazy 默认情况下,v-model 同步输入框的值和数据。可以通过这个修饰符,转变为在 change 事件再同步。

.number 自动将用户的输入值转化为数值类型

.trim 自动过滤用户输入的首尾空格

v-text 和 v-html :用来更新 textContent 和输出真正的 html 结构

v-pre :主要用来跳过这个元素和它的子元素编译过程。

v-cloak :保持在元素上直到关联实例结束时进行编译。

v-once :关联的实例,只会渲染一次。之后的重新渲染,实例极其所有的子节点将被视为静态内容跳过,这可以用于优化更新性能。

// 自定义指令 v-focus
  directives: {
    focus: {
      // 指令的定义
      inserted: function(el) {
        el.focus();
      },
    },
  },

19、vue 插槽如何使用

// 比如新建一个<base-layout> 组件
<div class="container">
  <header>
    <slot name="header"></slot>
  </header>
  <main>
    <slot></slot>
  </main>
  <footer>
    <slot name="footer"></slot>
  </footer>
</div>
// 使用插槽
<base-layout>
  <template v-slot:header> // 或者 <template #header>
    <h1>Here might be a page title</h1>
  </template>

  <template v-slot:default>
    <p>A paragraph for the main content.</p>
    <p>And another one.</p>
  </template>

  <template v-slot:footer>
    <p>Here's some contact info</p>
  </template>
</base-layout>

// 作用域插槽,比如新建一个<current-user>组件
<span>
  <slot v-bind:user="user">
    {{ user.lastName }}
  </slot>
</span>
// 使用插槽
<current-user>
  <template v-slot:default="slotProps">
    {{ slotProps.user.firstName }}
  </template>
</current-user>
// 或者缩写
<current-user v-slot="slotProps">
  {{ slotProps.user.firstName }}
</current-user>

20、vue 单页应用优缺点

优点:Vue 的目标是通过尽可能简单的 API 实现响应的数据绑定和组合的视图组 件,核心是一个响应的数据绑定系统。MVVM、数据驱动、组件化、轻量、简洁、高效、 快速、模块友好。 只关心数据,不关心 DOM。插件众多。维护状态方便。

缺点:不支持低版本的浏览器,最低只支持到 IE9;不利于 SEO 的优化(如果要支持 SEO,建议通过服务端来进行渲染组件);第一次加载首页耗时相对长一些;不可以使用浏 览器的导航按钮需要自行实现前进、后退。


21、为什么做 SSR,如何实现

更好的 SEO,由于搜索引擎爬虫抓取工具可以直接查看完全渲染的页面。

更快的内容到达时间,特别是对于缓慢的网络情况或运行缓慢的设备。通常 可以产生更好的用户体验。

开发条件所限。浏览器特定的代码,只能在某些生命周期钩子函数中使用; 一些外部扩展库可能需要特殊处理,才能在服务器渲染应用程序中运行。

涉及构建设置和部署的更多要求。与可以部署在任何静态文件服务器上的完 全静态单页面应用程序 (SPA) 不同,服务器渲染应用程序,需要处于 Node.js server 运行环境

更多的服务器端负载

通过后端返回 html 结构,在前端进行渲染展示,可使用 Nuxt 实现。


22、如何实现路由懒加载

为给客户更好的客户体验,首屏组件加载速度更快一些,解决白屏问题。

可以将页面进行划分,需要的时候加载页面,可以有效的分担首页所承担的加载压力

减少首页加载用时。

component: () => import("./Foo.vue");
// 或者
component: (resolve) => require(["@/components/home"], resolve);

23、less 如何设置全局样式

// vue.config.js 中
function addStyleResource(rule) {
  rule
    .use("style-resource")
    .loader("style-resources-loader")
    .options({
      patterns: [path.resolve(__dirname, "./src/assets/less/global.less")],
    });
}
// module.export 中 使用less
  css: {
    loaderOptions: {
      less: {
        lessOptions: {
          javascriptEnabled: true,
          globalVars: {
            primary: "#fff",
          },
        },
      },
    },
  },
// less全局变量
  chainWebpack: (config) => {
    const types = ["vue-modules", "vue", "normal-modules", "normal"];
    types.forEach((type) =>
      addStyleResource(config.module.rule("less").oneOf(type))
    );
  },

24、scoped 的作用是什么

scoped 的意思是下面的样式的作用域就是当前这个组件,不会影响全局的样式


25、$router和$route 的区别

$router是路由跳转,$route 是路由信息


26、data:{}和 data(){return {}} 的区别

因为不使用 return 包裹的数据会在项目的全局可见,会造成变量污染,使用 return 包裹后数据中变量只在当前组件中生效,不会影响其他组件。起到保护源数据的效果。


27、axios 的配置、封装、拦截和跨域

// 跨域:在config文件夹中的index.js中的devServe中填写
proxyTable:{ // cli3.0+  是在vue.config.js中配置proxy
 '/api':{
  Target:’代理服务器的目标地址’,
  changeOrigin: true,
  PathRewrite: {^/api”:” ”}
 }
}
// 封装和拦截
import axios from "axios";

const http = axios.create({
  baseURL: "/api",
  timeout: 5000,
  headers: {
    "Context‐Type": "application/json",
  },
});
// 请求拦截
http.interceptors.request.use(
  (res) => {
    // const token = sessionStorage.getItem('token') ? sessionStorag.getItem('token') : '';
    // if(token){给headers添加token}
    return res;
  },
  (err) => {
    return err;
  }
);
// 响应拦截
http.interceptors.response.use(
  (res) => {
    // const code = res.code
    // if(code === 404){router.replace()}
    // if(code === 200){router.replace()}
    return res;
  },
  (err) => {
    return err;
  }
);

function get(url, params = {}) {
  return new Promise((resolve, reject) => {
    http
      .get(url, params)
      .then((res) => resolve(res))
      .catch((err) => reject(err));
  });
}

function post(url, params = {}) {
  return new Promise((resolve, reject) => {
    http
      .post(url, params)
      .then((res) => resolve(res))
      .catch((err) => reject(err));
  });
}

export { get, post };

28、cli 各版本构建项目的命令和启动命令

创建项目:
cli2.0:vue init webpack 项目名
cli3.0+:vue create 项目名
启动项目:
cli2.0:npm run dev
cli3.0+:npm run serve


29、简单谈谈你对 vue3.0 的理解

29.1、vue3.0 和 vue2.0 最大的区别就是 api 从原来的 options API 变成了 composition API + options API ,编写代码更灵活、复用率更高。

29.2、vue3.0 比 vue2.0 快 2 倍,Tree-shaking 更友好

29.3、vue3.0 支持 TypeScript 以及 PWA

29.4、数据双向绑定从 Object.defineProperty 变成了 new Proxy,不用再使用$set 了

29.5、其他方面的更改:支持自定义渲染器、支持 Fragment 和 Protal 组件等。


30、简单说说双向绑定的原理

// 发布订阅模式实现的数据依赖采集器
class Dep {
  constructor() {
    this.subs = [];
  }
  addSub(sub) {
    this.subs.push(sub);
  }
  notify() {
    this.subs.forEach((sub) => sub.update());
  }
}
Dep.depTargets = null;

class Watcher {
  constructor(data, getter) {
    this.getter = getter;
    this.value = this.get();
  }
  get() {
    Dep.depTargets = this;
    let value = this.getter();
    Dep.depTargets = null;
    return value;
  }
  update() {
    this.value = this.get();
  }
}

const typeTo = (val) => Object.prototype.toString.call(val);

// 观察者模式监听所有属性的变化
function defineReactive(obj, key, value) {
  let dep = new Dep();
  Object.defineProperty(obj, key, {
    set(newValue) {
      if (newValue === value) return;
      value = newValue;
      dep.notify();
    },

    get() {
      const topTarget = Dep.depTargets;
      dep.addSub(topTarget);
      return value;
    },
  });
}

function walk(obj) {
  //监听所有属性
  Object.keys(obj).forEach((key) => {
    if (typeTo(obj[key]) === "[object Object]") {
      walk(obj[key]);
    }
    defineReactive(obj, key, obj[key]);
  });
}

function observe(value) {
  if (typeTo(value) !== "[object Object]") return null;
  walk(value);
}

observe(this.data);
new Watcher(this.data, () => {
  this.$mounte(this.el);
});

31、做过哪些 vue 的性能优化

31.1、长列表性能优化:可以通过 Object.freeze 方法来冻结一个对象,一旦被冻结的对象就再也不能被修改了。

export default {
  data: () => ({
    books: [],
  }),
  async created() {
    const books = await axios.get("/api/books");
    this.books = Object.freeze(books);
  },
};

31.2、优化无限列表性能:如果应用存在非常长或者无限滚动的列表,那么需要采用“窗口化”的技术来优化性能,只需要渲染少部分区域的内容,减少重新渲染组件和创建 dom 节点的时间。window.requestAnimationFrame 方法可以设置延迟加载的功能

setTimeout(() => {
  // 插入十万条数据
  const total = 100000;
  // 一次20条,可根据性能问题自己调整
  const MAX_ONCE = 20;
  // 渲染数据需要的次数
  const loopCount = total / MAX_ONCE;
  let countOfRender = 0;
  let el = document.querySelector("ul");
  function add() {
    // 优化,不允许插入数据引起回流
    const fragment = document.createDocumentFragment();
    for (let i = 0; i < MAX_ONCE; i++) {
      const li = document.createElement("li");
      li.innerText = `${i} + ${Math.floor(Math.random() * total)}`;
      fragment.appendChild(li);
    }
    el.appendChild(fragment);
    countOfRender += 1;
    loop();
  }

  function loop() {
    if (countOfRender < loopCount) {
      window.requestAnimationFrame(add);
    }
  }
  loop();
}, 0);

32、vue 的 diff 算法和虚拟 dom

虚拟 DOM 就是为了解决浏览器性能问题而被设计出来的。若一次操作中有 N 次更新 DOM 的动作,虚拟 DOM 不会立即操作 DOM,而是将这 N 次更新的 diff 内容保存到本地一个 JS 对象中,最终将这个 JS 对象一次性添加到 DOM 树上,再进行后续操作,避免大量无谓的计算量。所以,用 JS 对象模拟 DOM 节点的好处是,页面的更新可以先全部反映在 JS 对象(虚拟 DOM)上,操作内存中的 JS 对象的速度显然要更快,等更新完成后,再将最终的 JS 对象映射成真实的 DOM,交由浏览器去绘制。

vue 的 diff 算法:只进行同层级比较,忽略跨级操作,从两头向中间进行对比。

//diff时调用patch函数,patch接收两个参数vnode,oldVnode,分别代表新旧节点。
function patch (oldVnode, vnode) {
    if (sameVnode(oldVnode, vnode)) {
        patchVnode(oldVnode, vnode)
    } else {
        const oEl = oldVnode.el
        let parentEle = api.parentNode(oEl)
        createEle(vnode)
        if (parentEle !== null) {
            api.insertBefore(parentEle, vnode.el, api.nextSibling(oEl))
            api.removeChild(parentEle, oldVnode.el)
            oldVnode = null
        }
    }
    return vnode
}
// patch函数内第一个if判断sameVnode(oldVnode, vnode)就是判断这两个节点是否为同一类型节点
function sameVnode(oldVnode, vnode){
  //两节点key值相同,并且sel属性值相同,即认为两节点属同一类型,可进行下一步比较
    return vnode.key === oldVnode.key && vnode.sel === oldVnode.sel
}
//也就是说,即便同一个节点元素比如div,他的className不同,Vue就认为是两个不同类型的节点,执行删除旧节点、插入新节点操作。这与react diff实现是不同的,react对于同一个节点元素认为是同一类型节点,只更新其节点上的属性。

//对于同类型节点调用patchVnode(oldVnode, vnode)进一步比较
patchVnode (oldVnode, vnode) {
    const el = vnode.el = oldVnode.el  //让vnode.el引用到现在的真实dom,当el修改时,vnode.el会同步变化。
    let i, oldCh = oldVnode.children, ch = vnode.children
    if (oldVnode === vnode) return  //新旧节点引用一致,认为没有变化
    //文本节点的比较
    if (oldVnode.text !== null && vnode.text !== null && oldVnode.text !== vnode.text) {
        api.setTextContent(el, vnode.text)
    }else {
        updateEle(el, vnode, oldVnode)
        //对于拥有子节点(两者的子节点不同)的两个节点,调用updateChildren
        if (oldCh && ch && oldCh !== ch) {
            updateChildren(el, oldCh, ch)
        }else if (ch){  //只有新节点有子节点,添加新的子节点
            createEle(vnode) //create el's children dom
        }else if (oldCh){  //只有旧节点内存在子节点,执行删除子节点操作
            api.removeChildren(el)
        }
    }
}
// 更新vnode:updateChildren
function updateChildren (parentElm, oldCh, newCh, insertedVnodeQueue, removeOnly) {
    var oldStartIdx = 0;
    var newStartIdx = 0;
    var oldEndIdx = oldCh.length - 1;
    var oldStartVnode = oldCh[0];
    var oldEndVnode = oldCh[oldEndIdx];
    var newEndIdx = newCh.length - 1;
    var newStartVnode = newCh[0];
    var newEndVnode = newCh[newEndIdx];
    var oldKeyToIdx, idxInOld, vnodeToMove, refElm;

    // removeOnly is a special flag used only by <transition-group>
    // to ensure removed elements stay in correct relative positions
    // during leaving transitions
    var canMove = !removeOnly;

    {
      checkDuplicateKeys(newCh);
    }
    // 如果索引正常
    while (oldStartIdx <= oldEndIdx && newStartIdx <= newEndIdx) {
        // 当前的开始旧节点没有定义,进入下一个节点
      if (isUndef(oldStartVnode)) {
        oldStartVnode = oldCh[++oldStartIdx]; // Vnode has been moved left
        // 当前的结束旧节点没有定义,进入上一个节点
      } else if (isUndef(oldEndVnode)) {
        oldEndVnode = oldCh[--oldEndIdx];
        // 如果旧的开始节点与新的开始节点相同,则开始更新该节点,然后进入下一个节点
      } else if (sameVnode(oldStartVnode, newStartVnode)) {
     // 更新节点
        patchVnode(oldStartVnode, newStartVnode, insertedVnodeQueue);
        oldStartVnode = oldCh[++oldStartIdx];
        newStartVnode = newCh[++newStartIdx];
        // 如果旧的结束节点与新的结束节点相同,则开始更新该节点,然后进入下一个节点
      } else if (sameVnode(oldEndVnode, newEndVnode)) {
        patchVnode(oldEndVnode, newEndVnode, insertedVnodeQueue);
        oldEndVnode = oldCh[--oldEndIdx];
        newEndVnode = newCh[--newEndIdx];
        // 如果旧的开始节点与新的结束节点相同,更新节点后把旧的开始节点移置节点末尾
      } else if (sameVnode(oldStartVnode, newEndVnode)) { // Vnode moved right
        patchVnode(oldStartVnode, newEndVnode, insertedVnodeQueue);
        canMove && nodeOps.insertBefore(parentElm, oldStartVnode.elm, nodeOps.nextSibling(oldEndVnode.elm));
        oldStartVnode = oldCh[++oldStartIdx];
        newEndVnode = newCh[--newEndIdx];
        // 如果旧的结束节点与新的开始节点相同,更新节点后把旧的结束节点移置节点开头
      } else if (sameVnode(oldEndVnode, newStartVnode)) { // Vnode moved left
        patchVnode(oldEndVnode, newStartVnode, insertedVnodeQueue);
        canMove && nodeOps.insertBefore(parentElm, oldEndVnode.elm, oldStartVnode.elm);
        oldEndVnode = oldCh[--oldEndIdx];
        newStartVnode = newCh[++newStartIdx];
      } else {
          // 如果旧的节点没有定义key,则创建key
        if (isUndef(oldKeyToIdx)) { oldKeyToIdx = createKeyToOldIdx(oldCh, oldStartIdx, oldEndIdx); }
        idxInOld = isDef(newStartVnode.key)
          ? oldKeyToIdx[newStartVnode.key]
          : findIdxInOld(newStartVnode, oldCh, oldStartIdx, oldEndIdx);
        // 如果没有定义index,则创建新的新的节点元素
        if (isUndef(idxInOld)) { // New element
          createElm(newStartVnode, insertedVnodeQueue, parentElm, oldStartVnode.elm);
        } else {
          vnodeToMove = oldCh[idxInOld];
          if (sameVnode(vnodeToMove, newStartVnode)) {
            patchVnode(vnodeToMove, newStartVnode, insertedVnodeQueue);
            oldCh[idxInOld] = undefined;
            canMove && nodeOps.insertBefore(parentElm, vnodeToMove.elm, oldStartVnode.elm);
          } else {
            // same key but different element. treat as new element
            createElm(newStartVnode, insertedVnodeQueue, parentElm, oldStartVnode.elm);
          }
        }
        newStartVnode = newCh[++newStartIdx];
      }
    }
    // 如果旧节点的开始index大于结束index,则创建新的节点  如果新的开始节点index大于新的结束节点则删除旧的节点
    if (oldStartIdx > oldEndIdx) {
      refElm = isUndef(newCh[newEndIdx + 1]) ? null : newCh[newEndIdx + 1].elm;
      addVnodes(parentElm, refElm, newCh, newStartIdx, newEndIdx, insertedVnodeQueue);
    } else if (newStartIdx > newEndIdx) {
      removeVnodes(parentElm, oldCh, oldStartIdx, oldEndIdx);
    }
  }

33、vuex 页面刷新后数据丢失?和 history 模式刷新 404 问题?

//   在页面加载时读取sessionStorage里的状态信息
if (sessionStorage.getItem("state")) {
  this.$store.replaceState(
    Object.assign(
      {},
      this.$store.state,
      JSON.parse(sessionStorage.getItem("state"))
    )
  );
}

//   页面刷新时将state数据存储到sessionStorage中
window.addEventListener("beforeunload", () => {
  sessionStorage.setItem("state", JSON.stringify(this.$store.state));
});

// history模式刷新404问题   在vue.config.js中配置
module.exports = {
  publicPath: "/", //这个必须,引入静态资源需要从根路径引入,否则会找不到静态资源
  devServer: {
    // history模式下的url会请求到服务器端,但是服务器端并没有这一个资源文件,就会返回404,所以需要配置这一项
    historyApiFallback: {
      index: "/index.html", //与output的publicPath
    },
  },
};

34、Vue 开发中如何使用全局状态常量?你都用这个状态常量做什么事情

比如最常见的全局状态常量就是 process.env.NODE_ENV

它的值可能是:production、development

就是 webpack.config.js 中的 mode。

 {
  mode: "",
  entry: "",
  output: {}
 }

 // 比如现在是开发模式,我就显示一个某某功能按钮。
<button v‐if="process.env.NODE_ENV === 'develpment'">
测试按钮
</button>
npm run serve 的时候看的见这个按钮
npm run build 的时候看不见这个按钮

35、动态路由

通过 addRoutes() 动态添加路由信息


六、React

1、调用 setState 之后发生了什么?

在代码中调用 setState 函数之后,React 会将传入的参数对象与组件当前的状态合 并,然后触发所谓的调和过程(Reconciliation)。经过调和过程,React 会以相对高效的 方式根据新的状态构建 React 元素树并且着手重新渲染整个 UI 界面。在 React 得到元素 树之后,React 会自动计算出新的树与老树的节点差异,然后根据差异对界面进行最小化重 渲染。在差异计算算法中,React 能够相对精确地知道哪些位置发生了改变以及应该如何改 变,这就保证了按需更新,而不是全部重新渲染。


2、react 生命周期函数

初始化阶段:

getDefaultProps:获取实例的默认属性

getInitialState:获取每个实例的初始化状态

componentWillMount:组件即将被装载、渲染到页面上

render:组件在这里生成虚拟的 DOM 节点

componentDidMount:组件真正在被装载之后

运行中状态:

componentWillReceiveProps:组件将要接收到属性的时候调用

shouldComponentUpdate:组件接受到新属性或者新状态的时候(可以返回 false, 接收数据后不更新,阻止 render 调用,后面的函数不会被继续执行了)

componentWillUpdate:组件即将更新不能修改属性和状态

render:组件重新描绘 componentDidUpdate:组件已经更新

销毁阶段:

componentWillUnmount:组件即将销毁


3、为什么虚拟 dom 会提高性能?

虚拟 dom 相当于在 js 和真实 dom 中间加了一个缓存,利用 dom diff 算法避免了没 有必要的 dom 操作,从而提高性能。 用 JavaScript 对象结构表示 DOM 树的结构;然后用这个树构建一个真正的 DOM 树,插到文档当中当状态变更的时候,重新构造一棵新的对象树。然后用新的树和旧的树进 行比较,记录两棵树差异把 2 所记录的差异应用到步骤 1 所构建的真正的 DOM 树上,视 图就更新了。


4、react diff 原理

把树形结构按照层级分解,只比较同级元素。 给列表结构的每个单元添加唯一的 key 属性,方便比较。 React 只会匹配相同 class 的 component(这里面的 class 指的是组件的名字) 合并操作,调用 component 的 setState 方法的时候, React 将其标记为 dirty.到每一 个事件循环结束, React 检查所有标记 dirty 的 component 重新绘制. 选择性子树渲染。开发人员可以重写 shouldComponentUpdate 提高 diff 的性能。


5、React 中 refs 的作用是什么?

Refs 是 React 提供给我们的安全访问 DOM 元素或者某个组件实例的句柄。我们可以 为元素添加 ref 属性然后在回调函数中接受该元素在 DOM 树中的句柄,该值会作为回调函 数的第一个参数返回


6、展示组件和容器组件之间有何不同?

展示组件关心组件看起来是什么。展示专门通过 props 接受数据和回调,并且几乎不 会有自身的状态,但当展示组件拥有自身的状态时,通常也只关心 UI 状态而不是数据的状 态。 容器组件则更关心组件是如何运作的。容器组件会为展示组件或者其它容器组件提供数 据和行为(behavior),它们会调用 actions,并将其作为回调提供给展示组件。容器组件经 常是有状态的,因为它们是(其它组件的)数据源。


7、类组件和函数式组件之间有何不同?

类组件不仅允许你使用更多额外的功能,如组件自身的状态和生命周期钩子,也能使组 件直接访问 store 并维持状态 当组件仅是接收 props,并将组件自身渲染到页面时,该组件就是一个 ‘无状态组件 (stateless component)’,可以使用一个纯函数来创建这样的组件。这种组件也被称为哑组 件(dumb components)或展示组件


8、createElement 和 cloneElement 有什么区别?

React.createElement():JSX 语法就是用 React.createElement()来构建 React 元素 的。它接受三个参数,第一个参数可以是一个标签名。如 div、span,或者 React 组件。 第二个参数为传入的属性。第三个以及之后的参数,皆作为组件的子组件。

React.createElement(type, [props], [...children]);

React.cloneElement()与 React.createElement()相似,不同的是它传入的第一个参数 是一个 React 元素,而不是标签名或组件。新添加的属性会并入原有的属性,传入到返回 的新元素中,而旧的子元素将被替换。

React.cloneElement(element, [props], [...children]);

9、简述 flux 思想

Flux 的最大特点,就是数据的”单向流动”。 用户访问 View ,View 发出用户的 Action, Dispatcher 收到 Action,要求 Store 进行相应的更新, Store 更新后,发出一个”change”事件, View 收到”change”事件后,更新页面


10、了解 redux 么,说一下 redux

redux 是一个应用数据流框架,主要是解决了组件间状态共享的问题,原理是集中式管 理,主要有三个核心方法,action,store,reducer,工作流程是 view 调用 store 的 dispatch 接收 action 传入 store,reducer 进行 state 操作,view 通过 store 提供的 getState 获取最新的数据,flux 也是用来进行数据操作的,有四个组成部分 action, dispatch,view,store,工作流程是 view 发出一个 action,派发器接收 action,让 store 进行数据更新,更新完成以后 store 发出 change,view 接受 change 更新视图。 Redux 和 Flux 很像。主要区别在于 Flux 有多个可以改变应用状态的 store,在 Flux 中 dispatcher 被用来传递数据到注册的回调事件,但是在 redux 中只能定义一个可更新状态 的 store,redux 把 store 和 Dispatcher 合并,结构更加简单清晰

新增 state,对状态的管理更加明确,通过 redux,流程更加规范了,减少手动编码量, 提高了编码效率,同时缺点时当数据更新时有时候组件不需要,但是也要重新绘制,有些影 响效率。一般情况下,我们在构建多交互,多数据流的复杂项目应用时才会使用它们


11、什么是 JSX?

JSX 即 JavaScript XML。一种在 React 组件内部构建标签的类 XML 语法。JSX 为 react.js 开发的一套语法糖,也是 react.js 的使用基础。React 在不使用 JSX 的情况下一样可以工作, 然而使用 JSX 可以提高组件的可读性,因此推荐使用 JSX。

优点:

1.允许使用熟悉的语法来定义 HTML 元素树;

2.提供更加语义化且移动的标签;

3.程序结构更容易被直观化;

4.抽象了 React Element 的创建过程;

5.可以随时掌控 HTML 标签以及生成这些标签的代码;

6.是原生的 JavaScript。


12、React Hooks 是什么

Hook 是一个特殊的函数,它可以让你“钩入” React 的特性。例如,useState 是允 许你在 React 函数组件中添加 state 的 Hook。如果你在编写函数组件并意识到需要向其添 加一些 state,以前的做法是必须将其它转化为 class。现在你可以在现有的函数组件中使 用 Hook。

hooks 优势:无需复杂的 DOM 结构,简洁易懂

他允许你在不写 class 的情况下操作 state 和 react 的其他特性。hooks 只是多了一种写 组件的方法,使编写一个组件更简单更方便,同时可以自定义 hook 把公共的逻辑提取出 来,让逻辑在多个组件之间共享。


13、class 组件有什么不足

1.生命周期臃肿,逻辑耦合;

2.逻辑难以复用;通过继承解决,不支持多继承;通过高阶组件解决,会增加额外的组 件嵌套;通过渲染属性解决,也会增加额外组件嵌套,层级臃肿

3.class 存在 this 指向问题;匿名函数解决,每次创建新的函数,子组件重复不必要渲 染;bind 解决,需要写很多跟逻辑状态无关的代码


七、小程序

1、简单描述下微信小程序的相关文件类型

微信小程序项目结构主要有四个文件类型

WXML(WeiXin Markup Language)是框架设计的一套标签语言,结合基础组件、 事件系统,可以构建出页面的结构。内部主要是微信自己定义的一套组件

WXSS (WeiXin Style Sheets)是一套样式语言,用于描述 WXML 的组件样式

js 逻辑处理,网络请求

json 小程序设置,如页面注册,页面标题及 tabBar

主要文件

app.json 必须要有这个文件,如果没有这个文件,项目无法运行,因为微信框架把这 个作为配置文件入口,整个小程序的全局配置。包括页面注册,网络设置,以及小程序的 window 背景色,配置导航条样式,配置默认标题,底部导航等

app.js 必须要有这个文件,没有也是会报错!但是这个文件创建一下就行 什么都不需 要写以后我们可以在这个文件中监听并处理小程序的生命周期函数、声明全局变量

app.wxss 可选,全局样式


2、简述微信小程序原理

微信小程序采用 JavaScript、WXML、WXSS 三种技术进行开发,本质就是一个单页面 应用,所有的页面渲染和事件处理,都在一个页面内进行,但又可以通过微信客户端调用原 生的各种接口微信的架构,是数据驱动的架构模式,它的 UI 和数据是分离的,所有的页面 更新,都需要通过对数据的更改来实现

小程序分为两个部分 webview 和 appService 。其中 webview 主要用来展现 UI , appService 有来处理业务逻辑、数据及接口调用。它们在两个进程中运行,通过系统层 JSBridge 实现通信,实现 UI 的渲染、事件的处理


3、小程序的双向绑定和 vue 哪里不一样

小程序直接 this.data 的属性是不可以同步到视图的,必须调用: this.setData({ // 这里设置 })


4、小程序页面间有哪些传递数据的方法

4.1、使用全局变量实现数据传递,在 app.js 文件中定义全局变量 globalData, 将需要存 储的信息存放在里面

4.2、使用 wx.navigateTo 与 wx.redirectTo 的时候,可以将部分数据放在 url 里面,并 在新页面 onLoad 的时候接收

4.3、使用本地存储 Storage


5、小程序的生命周期函数

onLoad 页面加载时触发。一个页面只会调用一次,可以在 onLoad 的参数中获取打 开当前页面路径中的参数

onShow() 页面显示/切入前台时触发

onReady() 页面初次渲染完成时触发。一个页面只会调用一次,代表页面已经准备妥 当,可以和视图层进行交互

onHide() 页面隐藏/切入后台时触发。 如 navigateTo 或底部 tab 切换到其他页面, 小程序切入后台等

onUnload() 页面卸载时触发。如 redirectTo 或 navigateBack 到其他页面时


6、小程序组件通信

父传子:组件上写自定义属性,子组件通过 properties 接收

//父组件
<banner  step="{{step}}"></banner>

//子组件
Component({
  properties: {
    step:{
      type:Number
    }
  }
)},

子传父:通过自定义事件 triggerEvent

//子组件
 realNameConfirm(){
  let step=2;
     this.triggerEvent('realNameConfirm', step)     //通过triggerEvent将参数传给父组件
}

//父组件
<realName  bind:realNameConfirm="realNameConfirm" ></realName>

调取子组件的实例 selectComponent

<banner id="banner"></banner>;
this.selectComponent("#banner");

八、常见编程题

1、数组去重

// indexOf
function unique(array) {
  var newArr = [];
  for (var i = 0; i < array.length; i++) {
    if (newArr.indexOf(array[i]) === -1) {
      newArr.push(arr[i]);
    }
  }
  return newArr;
}

// 双层循环
function unique(array) {
  var r = [];
  for (var i = 0, l = array.length; i < l; i++) {
    for (var j = i + 1; j < l; j++) if (array[i] === array[j]) j = ++i;
    r.push(array[i]);
  }
  return r;
}

// hash表
function unique(array) {
  var hash = {};
  var arr1 = [];
  for (var i = 0; i < array.length; i++) {
    if (!hash[array[i]]) {
      hash[array[i]] = true;
      arr1.push(array[i]);
    }
  }
  return arr1;
}

// set
function unique(array) {
  return Array.from(new Set(array));
}

// 排序后,相邻相同去重
function unique(array) {
  array.sort();
  var re = [array[0]];
  for (var i = 1; i < array.length; i++) {
    if (array[i] !== re[re.length - 1]) {
      re.push(array[i]);
    }
  }
  return re;
}

2、数组排序

// 冒泡排序
function bubbleSort(array) {
  for (let i = 0; i < array.length; i++) {
    for (let j = i + 1; j < array.length; j++) {
      if (array[i] > array[j]) {
        const temp = array[i];
        array[i] = array[j];
        array[j] = temp;
      }
    }
  }
  return array;
}

// 选择排序
function selectionSort(array) {
  for (let i = 0; i < array.length; i++) {
    let minIndex = i;
    for (let j = i + 1; j < array.length; j++) {
      if (array[j] < array[minIndex]) {
        minIndex = j;
      }
    }
    const temp = array[i];
    array[i] = array[minIndex];
    array[minIndex] = temp;
  }
  return array;
}

// 插入排序
function insertionSort(array) {
  let temp = 0;
  for (let i = 1; i < array.length; i++) {
    let j = i;
    temp = array[i];
    while (j > 0 && temp < array[j - 1]) {
      array[j] = array[j - 1];
      j--;
    }
    array[j] = temp;
  }
  return array;
}

// 归并去重
function mergeSort(array, lo, hi) {
  if (lo >= hi) return;

  const mid = Math.floor(lo + (hi - lo) / 2);
  mergeSort(array, lo, mid);
  mergeSort(array, mid + 1, hi);

  merge(array, lo, mid, hi);
}
function merge(array, lo, mid, hi) {
  let i = lo;
  let j = mid + 1;
  const aux = [];

  for (let k = lo; k <= hi; k++) {
    aux[k] = array[k];
  }

  for (let k = lo; k <= hi; k++) {
    if (i > mid) {
      array[k] = aux[j++];
    } else if (j > hi) {
      array[k] = aux[i++];
    } else if (aux[j] < aux[i]) {
      array[k] = aux[j++];
    } else {
      array[k] = aux[i++];
    }
  }
}

3、统计字符串中出现次数最多的字符

var str = "12342535646757765453423";
var res = str.split("").reduce((res, cur) => {
  res[cur] ? res[cur]++ : (res[cur] = 1);
  return res;
}, {});
console.log(res); // {1: 1, 2: 3, 3: 4, 4: 4, 5: 5, 6: 3, 7: 3}

4、手写数组 filter 方法

Array.prototype.guolv = function (fn) {
  var arr = [];
  for (let i = 0; i < this.length; i++) {
    if (fn(this[i])) arr.push(this[i]);
  }
  return arr;
};

5、手写数组 map 方法

// 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;
};

// 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;
};

6、函数柯里化

// 标准版
function currying(fn) {
  let args = [];
  return function _c(...newArgs) {
    if (newArgs.length) {
      args = [...args, ...newArgs];
      return _c;
    } else {
      return fn.apply(this, args);
    }
  };
}

let addCurry = currying(add);
function add() {
  return [...arguments].reduce((prev, curr) => prev + curr);
}
// 注意调用方式的变化
console.log(addCurry(1)(2)(3)(4, 5)());

// 简单版
function sum(n) {
  function fn(m) {
    n += m;
    return fn;
  }
  fn.toString = function () {
    return n;
  };
  return fn;
}

7、数组扁平化

// 二维数组转一维数组
let arr = [
  [0, 1],
  [2, 3],
  [4, 5],
];
let newArr = arr.reduce((pre, cur) => {
  return pre.concat(cur); // 合并pre 与 cur, 并返回一个新数组
}, []);
console.log(newArr);

// 多维数组转一维数组
var arr = [[[[1, 2, 3], 4, 5], 6], 7, [[[8]], 9, 10]];

function flatten(arr) {
  var result = [];
  function iterator(arr) {
    for (let i = 0; i < arr.length; i++) {
      if (Array.isArray(arr[i])) {
        iterator(arr[i]);
      } else {
        result.push(arr[i]);
      }
    }
  }
  iterator(arr);
  return result;
}

console.log(flatten(arr));

8、手写 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))
    );
  };
};

9、数组交集、差集

//交集
function intersection(arr1, arr2) {
  return arr1.filter((item) => arr2.includes(item));
}

//差集
function differencesection(arr1, arr2) {
  return arr1.filter((item) => !arr2.includes(item));
}

10、连续重复最长的项

function maxRepeat(arr) {
  var i = 0;
  var j = 1;

  var maxCount = 0;
  var maxChar = "";

  while (i < arr.length) {
    if (arr[i] != arr[j]) {
      if (j - i > maxCount) {
        maxCount = j - i;
        maxChar = arr[i];
      }
      i = j;
    }
    j++;
  }

  return {
    maxCount,
    maxChar,
  };
}

console.log(maxRepeat([1, 1, 1, 2, 2, 2, 2, 2, 3, 3, 3, 3, 4, 4, 4]));

11、手写 queryString

// {"a":1,"b":2} => "a=1&b=2"

var result = [];
for (var k in o) {
  result.push(k + "=" + o[k]);
}
result.join("&");

// "a=1&b=2" => {"a":1,"b":2}

var str = "a=1&b=2";
var arr = str.split("&");
var obj = {};
for (var i = 0; i < arr.length; i++) {
  var k = arr[i].split("=")[0];
  var v = arr[i].split("=")[1];
  obj[k] = v;
}

12、手写 splice 方法

Array.prototype.removeIndex = function (index) {
  let length = this.length;
  for (let i = index; i < length; i++) {
    this[i] = this[i + 1];
  }
  this[length - 1] = null;
  this.length--;
};

13、手写 call 方法

Function.prototype.myCall = function (context) {
  context = context || window;
  context.fn = this;
  const args = [...arguments].slice(1);
  const result = context.fn(...args);
  delete context.fn;
  return result;
};

14、时间复杂度和空间复杂度

常见时间复杂度

img

img


九、其他

1、webpack 的基本配置

1.1、entry(项目入口,打包的入口文件,一个字符串或者一个对象):entry:”./src/index.js”

1.2、output(出口文件,配置打包的结果,一个对象):

output: {
  filename: "build.js";
}

1.3、module(模块的处理):loader 的配置主要在 module.rules 中进行,例如:

module: {
  rules: [
    {
      test: /(\.jsx|\.js)/, //表示匹配规则,是一个正则表达式
      use: {
        //表示针对匹配文件将使用处理的loader
        loader: "babel-loader",
        options: {
          presets: ["es2015", "react"],
        },
      },
    },
  ];
}

1.4、plugin(loader 不能做的处理都能交给 plugin 来做),例如:

const CleanWebpackPlugin = require("clean-webpack-plugin");

{
  plugin: [
    new webpack.DefinePlugin({
      "process.env": {
        NODE_ENV: JSON.stringify("production"),
      },
    }),
    new CleanWebpackPlugin(["js"], {
      root: __dirname + "/stu/",
      verbose: true,
      dry: false,
    }),
  ];
}

1.5、其它常用属性:

devTool:打包后的代码和原始代码存在较大的差异,此选项控制是否生成以及如何生成 sourcemap

devServer:通过配置 devserver 选项,可以开启一个本地服务器(通常在这里配置跨域)

  devServer: {
    proxy: {
      '/api': {
        target: 'http://www.baidu.com/',
        pathRewrite: {'^/api' : ''},
        changeOrigin: true,     // target是域名的话,需要这个参数,
        secure: false,          // 设置支持https协议的代理
      }
    }
  }

watch:启用 watch 模式后,webpack 将持续监听热河已经解析文件的更改,开发时开启会很方便

watchoption:用来定制 watch 模式的选项

performance:打包后命令行如何展示性能提示,如果超过某个大小是警告还是报错

1.6、webpack 打包优化常用配置

1.6.1、减小打包体积:

使用 CommonsChunkPlugin 分离第三方包

entry: {
 vendor: ['babel-polyfill', "axios", "marked", "react", "react-dom", "react-router-dom"], // 第三方文件
 app: './src/main.js'
},
plugins: [
 new webpack.optimize.CommonsChunkPlugin({
  name: "vendor", // 当加载 vendor 中的资源的时候,把这些资源都合并到 vendor.js 文件中
  filename: "js/vendor.js",
  minChunks: Infinity,
 })
],

或者使用 externals 进行 cdn 网络托管:

  // index.html页面中引入
  <script src="https://cdn.bootcss.com/react/15.0.0/react-with-addons.min.js"></script>
  <script src="https://cdn.bootcss.com/react/15.0.0/react-dom.min.js"></script>
  <script src="https://cdn.bootcss.com/react-router/3.0.0/ReactRouter.min.js"></script>
  <script src="https://cdn.bootcss.com/redux/3.6.0/redux.min.js"></script>
  <script src="https://cdn.bootcss.com/react-redux/5.0.1/react-redux.min.js"></script>
  <script src="https://cdn.bootcss.com/history/4.5.0/history.min.js"></script>
// webpack中配置
externals: {
 'react': 'React',
 'react-dom': 'ReactDOM',
 'react-router': 'ReactRouter',
 'redux': 'Redux',
 'history': 'History'
},

// vue.config.js
  configureWebpack:{
    externals: {
      'vue': 'Vue',
      'element-ui': 'ELEMENT',
      'vue-router': 'VueRouter'
    },
  }

1.6.2、小图片优化,使用 url-loader,将小图转成 base64,防止小图太多请求次数太多

npm install -D url-loader
module: {
 rules: [{
  test: /\.(png|svg|jpg|gif)$/,
  use: [{
  loader: 'url-loader', // 优化小图片过多造成请求数太多
  options: {
   limit: 8192, // 如果图片小于 8192 bytes 就直接 base64 内置到模板,否则才拷贝
   outputPath: 'img/'
  }
 }]
},

1.6.3、压缩 css、js 和 html:

 // 压缩css
 rules: [
  {
   test: /\.css$/,
   use: ExtractTextPlugin.extract({
   fallback: "style-loader",
   //这个地方配置一个对象,添加一个属性进行压缩css文件
   use: {
    loader: 'css-loader',
    options: {
     minimize: true   // 配置minimize 值为true,压缩css 文件
    }
   }
  })
 },

// 压缩js
/*下载*/ npm install -D uglifyjs-webpack-plugin
/*引入*/ const UglifyJsPlugin = require('uglifyjs-webpack-plugin')
/*配置*/
plugins: [
 new UglifyJsPlugin(), // 压缩 JavaScript
],

//压缩html
/*下载*/ npm   install  -D   html-webpack-plugin
/*引入*/ const HtmlWebpackPlugin = require('html-webpack-plugin')
/*配置*/
plugins: [
 new HtmlWebpackPlugin({
  template: './index.html', // 把 index.html 也打包到 dist 目录中
        // 压缩 html,默认 false 不压缩
        minify: {
         collapseWhitespace: true, // 去除回车换行符以及多余空格
            removeComments: true, // 删除注释
        }
    }),
],

2、git 的常用命令

2.1、首次拉取项目:git clone 项目地址

2.2、https 方式的需要输入用户名和密码,需要长期保存信息的话则输入:git config –global credential.helper winstore

ssh 方式的需要生成 ssh 密钥然后在 github 上设置,生成密钥:ssh-keygen -t rsa -C

2.3、查看分支:git branch -a

2.4、切换分支:git checkout 分支名

2.5、拉取项目:git pull origin 分支名

2.6、添加管理:git add .

2.7、提交代码:git commit -m 说明

2.8、查看提交:git log

2.9、推送提交:git push -u origin 分支名

2.10、版本回退:git reset –hard 目标版本号 或者 git revert -n 版本号

或者 git reset –hard HEAD^ 退回到前一个版本

2.11、回退后 push 会报错,需要强制推送:git push -f

2.12、合并分支:git merge 分支名


3、项目的工作流程

3.1、产品经理出原型图、流程图

3.2、UI 设计根据原型图出设计稿

3.3、项目经理出开发规范、前后端任务分配

3.4、前后端任务细分(根据功能、模块进行任务划分)、估计工期

3.5、前端进行静态页面开发、后端进行数据库搭建

3.6、后端写接口、接口联调、前端调接口渲染

3.7、提交、测试、修改 bug

3.8、优化、打包、发布


4、项目中你都负责哪块

4.1、后台管理系统:主要使用 cli 脚手架进行项目搭建,使用 elementUI 进行页面的快速开发,负责的模块主要有登录页:登录验证、登录拦截、登录过期、登录异常处理;首页:echarts 制表,有饼状图、柱状图、折线图,图表一键存为图片,动态渲染图表等;系统管理:系统设置,设置管理系统的版式、样式、风格;角色设置,设置角色的权限、个人信息、密码修改等;订单管理、文章管理、公告管理:进行相关的增删改查等操作。

4.2、商城:主要有首页(搜素模块、轮播组件、九宫格组件、列表组件、上拉加载更多组件、loading 组件)、详细页(自定义标题栏组件、模态框组件、消息提示框组件)、购物车页(商品数量增加,总金额实时计算)、登录注册页(表单验证、本地存储、数据加密、防 XSS、CSRF 攻击)、个人中心(我的收藏、我的设置、我的订单)等等


5、平时是如何进行自己的技能提升的

比如:去阮一峰个人网站、廖雪峰微博、csdn 社区、哔哩哔哩等视频网站上进行学习,也有关注一些公众号(开课吧、网易、金渡、优就业等),经常看一些大神发的帖子和文章。


6、你还有什么想要了解的

6.1、咱们公司现在正在开发什么项目,周期是多久,用的是什么技术?

6.2、咱们公司项目中的技术是否总是一成不变的,是否会使用新技术?比如什么时候会考虑使用 vue3?

6.3、咱们公司是否注重员工平时的技能提升,是否会有一些技能上的分享或者培训?

6.4、人员配比是什么样的,有几个前端几个后端,平时是如何沟通的(用的 git、svn、还是其它可视化工具)?

6.5、公司的晋升机制是什么样的?

6.6、如果我能有幸入职,是先有一段熟悉项目的时间还是直接就开始干?

6.7、您觉得我本次面试有哪些不足,假如您给我打分的话,100 分您会打多少分?


7、你都封装过哪些组件和工具方法

7.1、展示组件:container、row、col、header、footer 等

7.2、列表组件:整体是一个左图右文、固定宽+自适应的布局,内容部分放入了插槽,提高了组件的扩展性。

7.3、菜单组件:因为菜单层级是未知的,有的可能有二级,有的可能有三级或者更多层级,所以这块是通过一个递归组件来实现的,组件自己调用自己,最后暴露一个 menuData 属性,通过 props 来接收,父组件通过在子组件上写 menuData 属性来传值。

7.4、loading 组件、加载更多组件、分页组件等等。

7.5、格式化方法:时间戳格式化(将时间戳转换成 yyyy-mm-dd 的形式)、金钱格式化(金额前边加¥符号,每隔 3 位一个逗号)、cookie 的方法(getCookies、setCookies、removeCookies、clear 等)、请求方法(get、put、post、delete、options 等)


8、自我介绍

面试官您好,我叫 XX,今年 XX 岁,毕业于 XXXX 大学,什么专业,上一家公司是在 XX 城市的 XXXX 公司任前端开发一职,从事前端开发工作也有 X 年了,日常工作主要负责静态页面的开发、添加交互效果、数据请求和渲染以及产品的迭代、更新与维护。擅长的技术栈有 HTML、CSS、JS、Jquery、Vue、微信小程序,目前也有自学 React 和 nodeJS。曾参与过多个项目的开发,如 XX 商城、XX 官网、XX 后台管理系统等,有一定的实际开发经验。今天来应聘贵公司的前端开发岗位,以上就是我的自我介绍,谢谢。


9、项目介绍

我最近刚做完一个 XX 后台管理系统,主要的技术栈有:技术框架使用的是 vue2.0,项目使用 vue-cli3.0 搭建的,UI 框架使用的是 Element-UI,相关第三方还有 echarts 实现数据可视化,layload 实现图片懒加载,swiper 实现轮播图功能,vue-router 使用的是 history 模式(或者 hash 模式),路由采用的是动态路由,当用户登录成功时会返回角色权限,然后根据角色权限去动态创建路由(或者字典查询动态添加路由),路由全局守卫验证用户是否登录以及登录是否过期,数据请求使用的是 axios,有进行请求拦截和响应拦截,并封装 get 请求和 post 请求,数据管理使用的 vuex,数据持久化使用的 vuex-persistedstate(不熟的也可以说 storage)。主要负责的功能有:登录功能(表单验证、密码加密、过期验证、错误次数验证、手机+验证码登录、账号密码登录、自动登录、第三方登录);图表功能/大屏数据展示(营业额、销量、浏览量、分享量等各种饼状图、柱状图、折线图,支持一键生成图片,图表自适应等功能);系统管理:角色管理、用户管理、个人设置。角色管理主要是修改用户权限的,只有管理员才可见,而修改按钮则是按钮级别的权限,只有超级管理员才能修改,用户管理主要是商户端查看自己的会员的,个人设置主要是用户修改自己的个人信息和密码的;商品管理(商品分类、商品增删改查,发布商品使用的 wangEditor 富文本编辑器);订单管理(订单删查);公告管理(公共增删改查)等。


10、项目中遇到过哪些问题,以及如何解决的?

动态路由:以前写路由都是在 router 里面去配的,但是很多时候路由不可能全都配出来,当项目较大时,路由都是动态的,特别是有权限的路由,所以这块就遇到了很多问题。最后查了很多资料,vue 官网,github 社区,csdn 社区都有查看,最终找到了两种解决方法:

1、动态创建路由:用户登录成功后会返回权限信息,然后根据权限信息动态创建路由,最后添加到 router 上,伪代码如下:

//返回的数据res中包含如下属性
{
 roles: ['a','c','e']
}
let roles = res.roles;
let routes = roles.map(item=>{
 path: '/' + item,
 name: item,
 component: ()=>import('/'+item)
})
router.addRoutes(routes);

2、字典查询:用个 js 文件存储路由总表,然后根据角色权限去查询符合条件的路由信息,最后添加到 router 上,伪代码如下:

//返回的数据res中包含如下属性
{
  roles: ["a", "c", "e"];
}
let roles = res.roles;
// 路由总表
import allRoutes from "./allRoutes";
let routes = allRoutes.filter((item) => roles.includes(item.role));
router.addRoutes(routes);

按钮级权限: 全局 hasPermission

一开始不会做,自己查了很多资料,找到一种解决办法,就是在全局定义了一个方法 hasPermission,然后配合 v-if,在用户登录成功后,获取用户的按钮权限(数组格式),存储到 store 中,在需要的按钮上使用即可,伪代码大致如下:

function hasPermission(permission){
 let btns = store.getters.btns;
 return btns.indexOf(permission) > -1
}
Vue.prototype.hasPermission = hasPermission;

<el-button v-if="hasPermission('role:add')">添加</el-button>
<el-button v-if="hasPermission('role:update')">修改</el-button>
<el-button v-if="hasPermission('role:delete')">删除</el-button>
<el-button v-if="hasPermission('role:confirm')">确定</el-button>
<el-button v-if="hasPermission('role:cancel')">取消</el-button>

后来在公司的分享会上,其他同事还分享可以使用 vue 的自定义指令 directive,伪代码大致如下:

Vue.directive('has', {
 bind: function (el, binding, vnode) {   
       let   btnPermissions = vnode.context.$route.meta.btnPermissions.split(","); 
       let state = binding.value;
       switch(state){
         ......
       }
 }
});
<el-button v-btn="add">添加</el-button>
<el-button v-btn="update">修改</el-button>
<el-button v-btn="delete">删除</el-button>
<el-button v-btn="confirm">确定</el-button>
<el-button v-btn="cancel">取消</el-button>

第三方登录:微信、QQ 一键登录

操作数据:有时页面开发较快,后台接口没有写好,自己就要模拟请求写一套假数据,模拟数据使用的是 mock,但是自己写好之后也渲染完了,请求后端真实的接口时,由于和假数据的数据结构不一致,这时就要去修改数据

webpack 优化:详见九、其他:1、webpack 的基本配置

单元测试:test-utils


12、未来规划?

半年内争取先把 react 以及周边的第三方都能熟练使用;一年内争取能独自开发项目;一年半年把各种底层原理研究明白;两年内争取有 react+nodeJS+MySQL 的实际项目经验;三年内达到全栈开发的能力;至于更远的目标暂时还没有想过,如果畅谈一下的话,我想可能会朝着项目经理的方向发展,走管理层,或者组建自己的团队,也开公司当老板等。



  转载请注明: 24K博客 高频面试题

 上一篇
Vue常见面试问题及解析(持续更新中····) Vue常见面试问题及解析(持续更新中····)
1、如何理解 MVVM 原理 M:Model,数据层对数据的处理; V:View,视图层,是 HMTL 显示页面; VM:ViewModel:业务逻辑层(一切 JS 可视为业务逻辑,比如表单按钮提交,自定义事件的注册和处理逻辑都在View
2020-05-26
下一篇 
Javascript数组方法集锦 Javascript数组方法集锦
1、Array.from()介绍Array.from()方法从一个类似数组或可迭代对象创建一个新的,浅拷贝的数组实例。 语法Array.form(arrayLike[, mapFn[, thisArg]]) 参数 arrayLike -&g
  目录