范文健康探索娱乐情感热点
投稿投诉
热点动态
科技财经
情感日志
励志美文
娱乐时尚
游戏搞笑
探索旅游
历史星座
健康养生
美丽育儿
范文作文
教案论文

Web前端企业面试题

  1、display和visibility的区别?
  答:
  一、display和visibility的相同与不同点
  1、相同点:display和visibility都有讲元素隐藏的意思
  2、不同点:display是元素隐藏,隐藏的元素不占文档流 元素用display:none隐藏 下面的元素
  而visibility隐藏的元素仍然占文档流 元素用visibility: hidden;隐藏 下面的元素
  二、display和visibility的属性值
  1、display  display:block;将元素设置为块级元素,独占一行,能设置宽高   display:inline-block;将元素设置为行内快显示,能设置宽高,但是不独占一行   display:none;将元素设置为隐藏,后面的元素能自动往上填充
  2、visibility  visibility:visible;元素可见,默认值    visibility:hidden;将元素设置成隐藏,隐藏的元素仍然占据文档流   visibility:collapse;只对table对象起作用,用在其他元素中表示隐藏   visibility:inherit;继承其父级元素样式
  2、for 循环in、of和forEach的区别?
  答:
  在开发过程中经常需要循环遍历数组或者对象,使用最多的方法forEach、for…in 、 for…of,整理一下他们的异同点
  for循环
  其实除了这三种方法以外还有一种最原始的遍历,自Javascript诞生起就一直用的 就是for循环,它用来遍历数组var arr = [1,2,3,4] for(var i = 0 ; i< arr.length ; i++){ console.log(arr[i]) }
  for循环中可以使用return、break等来中断循环
  结果:
  forEach
  对数组的每一个元素执行一次提供的函数(不能使用return、break等中断循环),不改变原数组,无返回值undefined。let arr = ["a", "b", "c", "d"] arr.forEach(function (val, idx, arr) { console.log(val + ", index = " + idx) // val是当前元素,index当前元素索引,arr数组 console.log(arr) })
  输出结果:a, index = 0 (4) ["a", "b", "c", "d"] b, index = 1 (4) ["a", "b", "c", "d"] c, index = 2 (4) ["a", "b", "c", "d"] d, index = 3 (4) ["a", "b", "c", "d"]
  for…in
  循环遍历的值都是数据结构的键值let obj = {a: "1", b: "2", c: "3", d: "4"} for (let o in obj) { console.log(o) //遍历的实际上是对象的属性名称 a,b,c,d console.log(obj[o]) //这个才是属性对应的值1,2,3,4 }
  总结一句: for in也可以循环数组但是特别适合遍历对象
  结果:
  for…of
  它是ES6中新增加的语法,用来循环获取一对键值对中的值
  循环一个数组 let arr = ["China", "America", "Korea"] for (let o of arr) { console.log(o) //China, America, Korea }
  循环一个普通对象(报错) let obj = {a: "1", b: "2", c: "3", d: "4"} for (let o of obj) { console.log(o) //Uncaught TypeError: obj[Symbol.iterator] is not a function }
  3、px、em、rem之间有什么区别,移动端响应式布局方式有哪些?
  答:
  px:像素(Pixel),相对长度单位。像素px是相对于显示器屏幕分辨率而言的 。--即分辨率不同的机型当前对应的距离可能发生变化。(1Inches=xPx)。
  所以相对于同一机型来说是一个绝对的长度单位。
  em:是相对长度单位。相对于当前对象内文本的字体尺寸。如当前对行内文本的字体尺寸未被人为设置,则相对于浏览器的默认字体尺寸 。
  以其父级元素为基准来变化长度。所以其算法不是一个固定的值。
  rem:是CSS3新增的一个相对单位(root em,根em) 。
  那么其好用在用可以直接设置HTML的font-size,然后在其子类都是以这个大小为基准变化的值。
  在移动设备上,我们常常设置
  此方式的目的是为了将当前屏幕(PC)分辨率的px转化成手机端的px,让px在手机上能够兼容此方式。
  因此,此时看见的12px与PC端的12px无异。但是,如果不写着个meta的话,那么,浏览器会一直以PC的视图来决定手机端的px的大小,让视图看起来变小了很多。
  那么,响应式页面则经常配合媒体查询media 来设置不同分辨率下手机的各种不同的配置机构,然后根据父级来变化所有的相对长度。
  4、Js中操作数组的方法有哪些?
  基础篇
  1.shift
  // 删除原数组的第一项,并返回删除元素的值,如果数组为空则返回undefinedvar a = [1, 2, 3, 4, 5] var b = a.shift()// a:[2,3,4,5] b:1
  2.unshift
  // 将参数添加到原数组开头,并返回数组的长度var a = [1, 2, 3, 4, 5] var b = a.unshift(-2, -1)// a;[-2,-1,1,2,3,4,5] b:7
  //注:此方法在ie6.0下
  // 测试的返回值始终为undefined,在firefox下测试的返回值为7,所以此方法不可靠
  // 一般需要用返回值时可用splice代替
  3.pop
  // 删除原数组的最后一项,并返回删除元素的值;如果数组为空则返回undefinedvar a = [1, 2, 3, 4, 5] var b = a.pop()// a:[1,2,3,4] b:5
  4.push
  // 将参数添加到原数组末尾,并返回数组的长度var a = [1, 2, 3, 4, 5] var b = a.push(6, 7)// a:[1,2,3,4,5,6,7] b:7
  5.concat
  // 返回一个新数组,是将参数添加到原数组中构成的var a = [1, 2, 3, 4, 5] var b = a.concat(6, 7)// a:[1,2,3,4,5] b:[1,2,3,4,5,6,7]
  6.splice // splice(start,deleteCount.val1,val2,…):从start开始删除deleteCount项 // 并从该位置起插入val1,val2,…var a = [1, 2, 3, 4, 5] var b = a.splice(2, 2, 7, 8, 9)// a:[1,2,7,8,9,5] b:[3,4]
  7.reverse // 将数组反序 var a = [1, 2, 3, 4, 5] var b = a.reverse()// a:[5,4,3,2,1] b:[5,4,3,2,1]
  8.sort(orderfunction) // 按指定的参数对数组进行排序 var a = [1, 2, 3, 4, 5] var b = a.sort()// a:[1,2,3,4,5] b:[1,2,3,4,5]
  9.slice(start,end) // 返回从原数组指定开始的下标到结束下标之间的项组成的新数组 // 不改变原数组 var a = [1, 2, 3, 4, 5] var b = a.slice(2, 5)// a:[1,2,3,4,5] b:[3,4,5]
  10.join(separator) // 将数组的元素组起一个字符串,以separator为分隔符,省略 // 的话则用默认的逗号为分隔符 var a = [1, 2, 3, 4, 5] var b = a.join("|")// a:[1,2,3,4,5] b:"1|2|3|4|5"
  11.indexOf // 数组元素索引并返回元素索引,不存在返回-1,索引从0开始,是从前往后查找 var a = ["a", "b", "c", "d", "e"] a.indexOf("a")// 0 a.indexOf(a)// -1 a.indexOf("f")// -1 a.indexOf("e")// 4
  12.lastIndexOf // 接受两个参数:查找的值、查找起始位置 // 不存在,返回 -1 ;存在,返回位置。是从后往前查找 var a= [2, 5, 9, 2] a.lastIndexOf(2) // 3 a.lastIndexOf(7) // -1 a.lastIndexOf(2, 3) // 3 a.lastIndexOf(2, 2) // 0 a.lastIndexOf(2, -2) // 0 a.lastIndexOf(2, -1) // 3
  13. Array.isArray() // 判断对象是否是数组,返回布尔值。不改变原数组。 var a = 1 var test = [1, 2, 3, 4] console.log(Array.isArray(a)) // false console.log(Array.isArray(test)) // true
  14. Array.toString() // 将数组转化为字符串,返回字符串,不改变原数组 var test = [1, 2, 3, 4] test1 = test.toString() console.log(test) // [1, 2, 3, 4] console.log(test1) // 1,2,3,4
  进阶篇
  1.forEach
  是最为常用的情景,它至于遍历,可以在获取当前数据项的前提下,对数据进行修改。它没有返回值。理解起来也是最容易的。let xxb = ["北京", "南京", "广州", "重庆", "西安", "拉萨"]; xxb.forEach(item=> { console.log(item,"forEach"); })
  2.map
  map的本意就是映射,也就是将一个值从一种形式映射到另一种形式,比如将key映射到value。它的每一次遍历都会有一个返回值。这些返回值组合成最终的结果数组。事实就是如此let xxb = ["北京", "南京", "广州", "重庆", "西安", "拉萨"]; let xxb_map = xxb.map((item,index,list)=> { console.log(item,"mapitem"); // console.log(index,"mapindex"); // console.log(list,"maplist"); // return item+"市" }) console.log(xxb_map,"58")
  forEach和map对比
  相同点
  1)都是循环遍历数组中的每一项;
  2)forEach()和map()匿名函数的参数相同,参数分别是item(当前每一项)、index(索引值)、arr(原数组);
  3)this都是指向调用方法的数组;
  4) 只能遍历数组;
  不相同点
  1)map()创建了新数组,不改变原数组;forEach()可以改变原数组。
  2)遇到空缺的时候map()虽然会跳过,但保留空缺;forEach()遍历时跳过空缺,不保留空缺。
  3)map()按照原始数组元素顺序依次处理元素;forEach()遍历数组的每个元素,将元素传给回调函数。
  3.filter
  它致力于从已有的数组中筛选出符合一定条件的数据项,最后的返回值是所有符合条件的数据项构成的数组。它不会修改原来的数组。记住,它的立足点就是筛选。也仅仅是筛选。还有一点需要注意:每一次遍历都会有一个返回值,它的类型是布尔类型。返回值只有是true,当前遍历项才会被筛选中。不要试图在filter中去修改原始数组。
  理解: 就是在一堆数据里面去筛选你需要的数据 或者 剔除你不需要的数据function City (province, school, level) { this.province = province; this.school = school; this.level = level; } let beijing = new City("北京都", ["北京大学", "清华大学"], 1); let xian = new City("西安都", ["西安交通大学", "西北工业大学"], 2); let hubei = new City("湖北省", ["武汉大学", "华中科技大学"], 2); let hunan = new City("湖南省", ["湖南大学", "中南大学"], 3); let sichuan = new City("四川省", ["四川大学", "电子科技大学"], 3); let cities = [beijing, xian, hubei, hunan, sichuan]; let tops = cities.filter(function(item, index, list) { return item.level != 1; // 筛选出 level 不为 1 的数组 }); console.log(tops, "77");
  4.find()
  返回通过测试的数组的第一个元素的值,
  理解:假如你去一个一群人的地方去找人 你说我找xxb 他要是在那里就找到 不在那里就是undefined
  在第一次调用 callback 函数时会确定元素的索引范围,因此在 find 方法开始执行之后添加到数组的新元素将不会被 callback 函数访问到。如果数组中一个尚未被callback函数访问到的元素的值被callback函数所改变,那么当callback函数访问到它时,它的值是将是根据它在数组中的索引所访问到的当前值。被删除的元素仍旧会被访问到。
  语法
  array.find(function(value, index, arr),thisValue)
  value:必须,代表当前元素,其他四个参数都是可选,index代表当前索引值,arr代表当前的数组,thisValue代表传递给函数的值,一般用this值,如果这个参数为空,undefined会传递给this值
  返回值:返回符合测试条件的第一个数组元素的值,如果没有符合条件的则返回undefined。var arr = [1,2,3,4,5,6,7]; var ar = arr.find(function(elem){ return elem>5; }); console.log(ar);//6 console.log(arr);//[1,2,3,4,5,6,7]
  扩展: findIndex()方法的用法与find()方法非常类似,返回第一个符合条件的数组成员的位置,如果所有成员都不符合条件,则返回-1。自己try吧
  5.every
  它地用武之地在哪里呢?不期望它能对原始数组进行修改。它地定位是考察数组地整体特性。也就是考察数组中所有元素的共性。比如所有元素是否都是奇数,或者所有元素是否都是偶数。它关注的是数组整体元素的共性。只要有一个不满足,循环就会结束,接下来的数据就不会继续判断。
  理解: 这个就像 上课了 老师说 昨天布置的作业 只要有一个人没有写完 今天就不讲课 然后挨个检查 当检查到有一个没有写完的时候 就不检查了 返回 不讲课了(false)// var numbers = [1, 2, 3, 4, 5, 6, 7, 8, 9]; var numbers = [2, 4,, 6, 8, 9,10]; let bool = numbers.every(function(item, index, list) { console.log(item,"90"); return item % 2 == 0; }); console.log(bool,"93");
  6.some
  它地用武之地在哪里呢?它的定位是考察数组的个性。比如考察数组中是否存在一个等于0的数。它关注的是数组的个性。只要有一个满足,循环就会结束,接下来的数据就不会继续判断。
  理解: 相当于在一个数组里面找你想要的那个数 找到了 就返回true 找不到就返回falsevar numbers = [2, 4, 6, 8, 9,10]; let bool = numbers.some(function(item, index, list) { console.log(item,97); return item == 4; }); console.log(bool,"100");
  7.reduce
  它这个方法是接收一个函数作为累加器,将数组中的值(从左向右)开始合并,最总为一个值 然后返回出来,callback
  他可以传四个参数:
  1,previousValue:上一次调用回调返回的值,或者是提供的初始值(initialValue)
  2,currentValue:数组中当前被处理的元素
  3,index:不啰嗦,自己顾名思义去吧
  4,array:返回调用reduce的数组// 获取购物车中商品列表的价格总和 let goodList = [{id: 1, price: 10, qty: 5}, {id: 2, price: 15, qty: 2}, {id: 3, price: 20, qty: 1}] let totalPrice = goodList.reduce((prev, cur, index, arr) => { // prev就是每次合并返回的值 第一次返回时还没有做合并就是返回的默认值0 // 下面分别是10*5=50; ---- 15*2=30+50=80;---- 20*1=20+80=100 console.log(prev, "----", cur, "----", index, "----", arr) return prev + cur.price * cur.qty // 这个0就是prev 的默认值 }, 0) console.log(totalPrice, "totalPrice")
  var arrString = "abcdaabc" // 获取字符中中每个字母出现的次数 let count = arrString.split("").reduce(function(res, cur) { console.log(res, cur, "res, cur") res[cur] ? res[cur]++ : res[cur] = 1 return res }, {}) console.log(count, "count") // {a: 3, b: 2, c: 2, d: 1}
  彩蛋: 这里附上前几天某某人比较喜欢的切割字符串的方法
  substring() 和 substr()
  相同点: 如果只是写一个参数,两者的作用都一样:都是是截取字符串从当前下标以后直到字符串最后的字符串片段。substr(startIndex) substring(startIndex) var str = "123456789" console.log(str.substr(2)) // "3456789" console.log(str.substring(2)) // "3456789"
  不同点: 第二个参数
  substr(startIndex, lenth) // 第二个参数是截取字符串的长度(从起始点截取某个长度的字符串);
  substring(startIndex, endIndex) // 第二个参数是截取字符串最终的下标 (截取2个位置之间的字符串,‘含头不含尾")。console.log("123456789".substr(2, 5)) // "34567" console.log("123456789".substring(2, 5)) // "345"
  5、Vue中created钩子和mounted钩子有什么区别?
  1:在使用vue框架的过程中,我们经常需要给一些数据做一些初始化处理,这时候我们常用的就是在created与mounted选项中作出处理。
  首先来看下官方解释,官方解释说created是在实例创建完成后呗立即调用。在这一步,实例已完成以下配置:数据观测 (data observer),属性和方法的运算,watch/event 事件回调。然而,挂载阶段还没开始,$el 属性目前不可见。
  这话的意思我觉得重点在于说挂架阶段还没开始,什么叫还没开始挂载,也就是说,模板还没有被渲染成html,也就是这时候通过id什么的去查找页面元素是找不到的。下面看下实例来证明。
  所以,一般creadted钩子函数主要是用来初始化数据。
  2:mounted钩子函数一般是用来向后端发起请求拿到数据以后做一些业务处理。官方解释如下:
  el 被新创建的 vm.$el 替换,并挂载到实例上去之后调用该钩子。如果 root 实例挂载了一个文档内元素,当 mounted 被调用时 vm.$el 也在文档内。
  这意思是该钩子函数是在挂在完成以后也就是模板渲染完成以后才会被调用。下面看实例
  下面是结果
  nihao
  取到了值,这说明这时候vue模板已经渲染完毕。因此,Dom操作一般是在mounted钩子函数中进行的
  computed:{} 计算属性,什么是计算属性呢,我个人理解就是对数据进行一定的操作,可以包含逻辑处理操作,对计算属性中的数据进行监控。计算属性是基于它的以来进行更新的,只有在相关依赖发生改变时侧能更新变化,以函数的形式返回结果。然后可以像绑定普通属性一样在模板中绑定计算属性。   {{msg}}  网址 {{msg}}的网络协议是:{{msg2}}    
  总结:
  通常created使用的次数多,而mounted通常是在一些插件的使用或者组件的使用中进行操作,比如插件chart.js的使用: var ctx = document.getElementById(ID);通常会有这一步,而如果你写入组件中,你会发现在created中无法对chart进行一些初始化配置,一定要等这个html渲染完后才可以进行,那么mounted就是不二之选。
  methods:{}中的方法都需要主动去触发,比如点击click之类的
  而created(){}、mounted(){}、里面的代码都是自动去执行的,即vue生命周期到了哪一步就直接去执行对应钩子函数里面的代码了,无需手动去执行
  created中主要放初始化获取数据之类,mounted()中挂载到具体的DOM节点
  computed:{} 计算属性,什么是计算属性呢,我个人理解就是对数据进行一定的操作,可以包含逻辑处理操作,对计算属性中的数据进行监控。计算属性是基于它的以来进行更新的,只有在相关依赖发生改变时侧能更新变化,以函数的形式返回结果。然后可以像绑定普通属性一样在模板中绑定计算属性。    {{msg}} 网址 {{msg}}的网络协议是:{{msg2}}   
  mounted 是生命周期钩子,vue的生命周期中一个实例的mounted只会运行一次。mounted在vue的渲染模板挂载到$el元素上才会调用,很显然你export的时候el都么有自然不会运行mounted钩子函数了。
  所有的方法都应该在methods里定义,然后在created或者mounted里 使用this调用方法,用这种方式实现初始化
  6、Vue中组件之间的传参方式有哪些方式?
  Vue 组件传参的八种方式总结
  Vue 组件的使用不管是在平常工作还是在面试面试中,都是频繁出现的。因此系统的梳理一下组件之间的传参还是非常有必要的
  一、props 传参
  子组件定义 props 有三种方式:
  // 第一种数组方式props: [xxx, xxx, xxx]
  // 第二种对象方式props: { xxx: Number, xxx: String}
  // 第三种对象嵌套对象方式props: { xxx: { //类型不匹配会警告 type: Number, default: 0, required: true, // 返回值不是 true,会警告 validator(val) { return val === 10} } }
  第三种对象默认支持 4 种属性,并且都是非必填的。可以随意使用
  父组件传参的俩种方式
  第一种静态属性传参
  注意:
  1、在不定义 props 类型的情况下 props 接受到的均为 String。
  2、当 props 属性指定为 Boolean 时,并且只有属性 key 没有值 value 时接受到的是 true   
  第二种动态属性传参
  注意:
  1、需要区分非简写形式传入的值是对象,则会对应 props 中多个值
  2、会保留传入值的类型
  3、如果是表达式则获取到的是表达式的计算结果     
  二、attrs 和listeners
  $attrs
  $attrs 会获取到 props 中未定义的属性(class 和 style 属性除外),支持响应式。常用的场景有俩种:
  组件嵌套组件时可以使用 $attrs 来支持过多的属性支持。比如 elementUI 的 table 组件。支持的属性十几个,而平常封装的时候用的最多的也就一俩个。
  属性默认是添加在父组件上的,有时候想把多余的属性添加在子组件上(可以结合 inheritAttrs: false 属性,让父属性不接受多余的属性)
  $listeners 定义的事件都在子组件的根元素上,有时候想加到其他元素上。就可以使用 $listerners。它包含了父组件中的事件监听器(除了带有 .native 修饰符的监听器)
  三、$emit 通知
  Vue 默认有 $on $emit $once $off 几种方法来实现发布订阅模式,这也应用在了组件传参上。在组件上添加的特殊方法 @abc="methods" 就相当于使用了 $on 来监听这个方法。因此组件内可以使用 $emit 来进行通知。
  这里有一道考题: for 循环的时候如何拿到子组件的传值和 for 中循环的值
  答案有俩种,一是 $event, 二是 闭包。只是需要注意 $event 只能获取到第一个值
  四、v-model
  这个其实是一种通过 emit,on 的组合方式。优点再于同步值方便,写法优雅。下面三种写法其实是一个意思// 写法 1  { model: { prop: "value", event: "update:a", }, methods: { a() { this.$emit("update:a", 1)} } } // 写法 2  { props: ["a"] methods: { a() { this.$emit("update:a", 1)} } } // 写法 3 // 1. 事件名必须是 update: + 属性名 // 2. 参数不能是表达式,最好是 data 里面的属性  { props: ["a"] methods: { a() { this.$emit("update:a", 1)} } }
  五、插槽     123             
  六、$refs, $root, $parent, $children
  $root 获取根组件
  $parent 获取父组件
  $children 获取子组件(所有的子组件,不保证顺序)
  $refs 组件获取组件实例,元素获取元素
  七、project / inject
  注意:注入的值是非响应的 { project() { return { parent: this } } }  { // 写法一 inject: ["parent"] // 写法二 inject: { parent: "parent" } // 写法三 inject: { parent: { from: "parent", default: 222 } } }
  八、Vuex
  这个相当于单独维护的一组数据,就不过多的说了。
  7、Vue中watch和computed之间有什么区别?
  watch,computed和methods的关系
  1.watch和computed都是以Vue的依赖追踪机制为基础的 ,它们都试图处理这样一件事情:当某一个数据(称它为依赖数据)发生变化的时候,所有依赖这个数据的"相关"数据"自动"发生变化,也就是自动调用相关的函数去实现数据的变动。
  2.对methods:methods里面是用来定义函数的,很显然,它需要手动调用才能执行。而不像watch和computed那样,"自动执行"预先定义的函数
  watch和computed各自处理的数据关系场景不同
  1.watch擅长处理的场景:一个数据影响多个数据
  2.computed擅长处理的场景:一个数据受多个数据影响
  watch用法 监听下记haiZeiTuan_Name的值,会改变其他所有的值 // 在watch中,一旦haiZeiTuan_Name(海贼团名称)发生改变 data选项中的船员名称全部会自动改变 (suoLong,naMei,xiangJiShi),并把它们打印出来 // 多个)船员名称数据 --> 依赖于 --> (1个)海贼团名称数据一个数据变化 ---> 多个数据全部变化 data: { haiZeiTuan_Name: "草帽海贼团", suoLong: "草帽海贼团索隆", (haiZeiTuan_Name + 海贼名称) naMei: "草帽海贼团娜美", xiangJiShi: "草帽海贼团香吉士" }, watch: { haiZeiTuan_Name: function (newName) { this.suoLong = newName + "索隆" this.naMei = newName + "娜美" this.xiangJiShi = newName + "香吉士" console.log(this.suoLong) console.log(this.naMei) console.log(this.xiangJiShi) } } // 更改watch选项中监控的主数据vm.haiZeiTuan_Name = "橡胶海贼团"
  结果:this.suoLong会变为 "橡胶海贼团索隆",以此类推
  computed用法 监听下记firstName,secName,thirdName的值,会改变luFei_Name的值 data: { // 路飞的全名:蒙奇·D·路飞 firstName: "蒙奇", secName: "D", thirdName: "路飞" }, computed: { luFei_Name: function () { return this.firstName + this.secName + this.thirdName } } // 将"路飞"改为"海军王" vm.thirdName = "海军王" // 蒙奇·D·海军王
  methods和computed的区别例子 new Vue({ el: "#app", // 设置两个button,点击分别调用getMethodsDate,getComputedDate方法 template: "   ", methods: { getMethodsDate: function () { alert(new Date()) }, // 返回computed选项中设置的计算属性——computedDate getComputedDate: function () { alert(this.computedDate) } }, computed: { computedDate: function () { return new Date() } }
  注意两次点击computed返回的时间是相同的!!
  1.两次点击methods返回的时间是不同的
  2.注意两次点击computed返回的时间是相同的
  【注意】为什么两次点击computed返回的时间是相同的呢?new Date()不是依赖型数据 (不是放在data等对象下的实例数据) ,所以computed只提供了缓存的值,而没有重新计算
  只有符合:1.存在依赖型数据 2.依赖型数据发生改变这两个条件 ,computed才会重新计算。
  参考:http://www.cnblogs.com/penghuwan/p/7194133.html
  8、在Vue项目中用到哪些性能优化问题?是这么处理的?
  答:https://codecat.blog.csdn.net/article/details/100031285
  9、async和promise的作用和用法分别有哪些?
  1、Promise
  Promise 是异步编程的一种解决方案,比传统的解决方案——回调函数和事件——更合理和更强大,简单地说,Promise好比容器,里面存放着一些未来才会执行完毕(异步)的事件的结果,而这些结果一旦生成是无法改变的
  2、async await
  async await也是异步编程的一种解决方案,他遵循的是Generator 函数的语法糖,他拥有内置执行器,不需要额外的调用直接会自动执行并输出结果,它返回的是一个Promise对象。
  两者的主要用法、语法就不赘述了,感兴趣的同学可以自行查阅 es6中文文档
  两者的区别
  1、Promise的出现解决了传统callback函数导致的"地域回调"问题,但它的语法导致了它向纵向发展行成了一个回调链,遇到复杂的业务场景,这样的语法显然也是不美观的。而async await代码看起来会简洁些,使得异步代码看起来像同步代码,await的本质是可以提供等同于"同步效果"的等待异步返回能力的语法糖,只有这一句代码执行完,才会执行下一句。
  2、async await与Promise一样,是非阻塞的。
  3、async await是基于Promise实现的,可以说是改良版的Promise,它不能用于普通的回调函数。
  简单来看,这两者除了语法糖不一样外,他们解决的问题、达到的效果是大同小异的,我们可以在不同的应用场景,根据自己的喜好来选择使用。// promise方法 let p1 = new Promise((resolve,reject) => { setTimeout(() => { resolve("我是p1") },4000) }) let p2 = new Promise((resolve,reject) => { setTimeout(() => { resolve("我是p2") },200) }) let p3 = new Promise((resolve,reject) => { setTimeout(() => { resolve("我是p3") },200) }) // 想让p1完成后再执行P2再执行P3 // 数量太多只能这么写 p1.then((res) => { console.log(res); p2.then((res) => { console.log(res); p3.then((res) => { console.log(res); }) }) }) //下面大神回答的挺好的 p1.then((res)=>{ console.log(res) return p2 }) .then((res)=>{ console.log(res) return p3 }) .then((res)=>{ console.log(res) }) // async await语法糖 let a1=()=>{ return new Promise((resolve,reject) => { setTimeout(() => { resolve("我是a1") },4000) }) } let a2=()=>{ return new Promise((resolve,reject) => { setTimeout(() => { resolve("我是a2") },40) }) } let a3=()=>{ return new Promise((resolve,reject) => { setTimeout(() => { resolve("我是a3") },40) }) } // 想让a1完成后再执行a2再执行a3 //能避免回调 async function asy(){ await a1().then((res) => {console.log(res)}); await a2().then((res) => {console.log(res)});; await a3().then((res) => {console.log(res)});; } asy();

30的女人,眼霜看这5款!实打实抗老紧致,款款都不容错过30女性的眼周肌肤所需护理,不单单要足够滋润保湿,更要关注到眼周肌肤老化问题。尽早对眼周皮肤进行抗衰护理,才能让脆弱的眼周皮肤保持在健康年轻态。市面上主攻抗老护肤的眼霜产品,在具体何超仪任性了!穿露脐装显得人好油腻,把整个人形象拉低了档次都说有钱可以任性,有的时候真的也是有几分道理的!尤其是在穿衣上,不见得越贵越有特色的,就越好看!相反的,还有可能会拉低你的形象和气质呢!像何超仪,出身于赌王家族,所以天生就自带了很孙悦腿伤彻底好了!戴礼帽脚踩1190元白鞋去森林,50岁还有少女感歌手孙悦凭借祝你平安红遍大江南北,事业也是达到了顶峰!她就在当红时退出歌坛嫁人,婚后的生活非常惬意,安心当全职妈妈过着阔太的生活,也会偶尔分享自己的日常,让大家知道她过得很好。沉寂今蝉祛痘科普痘痘反反复复,该怎么消除痘印?每个人的青春都会有浓墨重彩的一笔,痘痘也不例外,有些人在青春都会长青春痘,在青春时它是青春的证明,可当青春不再了,这些青春痘就会在脸上留下坑坑洼洼的痕迹,令人苦恼。再加上本身的肤质怀斯曼养伤时我曾因情绪低落而无法训练北京时间8月17日,据ClutchPoints报道,勇士球员詹姆斯怀斯曼在近期接受采访时表示,漫长的养伤的过程经常使自己情绪低落。詹姆斯怀斯曼由于膝伤错过了整个202122赛季。怀KLOCK智能锁一键开启幸福的懒人生活不知道大家有没有听过这样一个段子枯藤老树昏鸦空调wifi西瓜葛优同款沙发夕阳西下,我就往上一趴当段子成为当代年轻人的真实写照,懒人经济一词也走入大众视野因为懒得走路,于是发明了汽车6岁女童得怪病突然不会走路,专家其家中两个孩子都有这个情况,确认和基因有关据河南电视台民生频道近日,河南平顶山6岁女童佩琪患怪病,原本活泼爱动的她突然不会走路,且双脚会不停抽搐。马上到了该上学的年纪却突然不能奔跑,全家人都备受打击。记者现场触摸孩子的小腿吃瓜!阿杜终于赢了!格林真男人啊!三个孩子只有一个是亲生休赛期,追梦格林可没少忙活,与勇士因为续约问题双方扯来扯去,最后也没有个定性然后在开通了自己的播客,在上面与球迷球星唠起了家常。最后一件就是人生大事结婚!近日,格林与女友蕾妮修成正追梦格林娶了杜兰特初恋,这一次阿杜终于翻身了太意外了,这一次杜兰特和追梦格林的对决,终于扳回了一城。就在最近,追梦格林完成了人生中的大事,举行了自己的婚礼,这本是一件可喜可贺的事情,但是让大家有些意外的是,追梦格林所娶之人,10秒99!新星卢肯坎佩尔爆冷夺得欧洲冠军,强于亚洲百米水平2022年8月17日,慕尼黑欧洲锦标赛女子100米飞人大战拉开序幕。来自德国的新星吉娜卢肯坎佩尔以10秒984的成绩爆冷夺得欧锦赛百米冠军,仅仅以0。005秒的优势险胜坎邦德吉。而千亿车联网市场规模带给磁元件企业哪些新机哔哥哔特导读行业机构预测,2022年车联网市场规模达2771亿元,这又会给磁性元件行业带来怎样的发展机遇呢?近日,交通运输部发布自动驾驶汽车运输安全服务指南(试行)(征求意见稿),愤怒!乒坛名将驳斥无脑网友,否认欧洲球队不来中国参赛谣言北京时间8月17日,德国华裔乒乓球名将单晓娜正式发布视频辟谣,称自己之前在直播中说的话被一些无脑的网友扭曲并扩散,以至于许多人误以为欧洲的球队将不会参加即将在9月举办的成都团体世乒为什么我的头发会出这么多油?头发油可以说是生活中很常见的一个情况,也会让人们感觉到头痛,特别是女性对头发出油的敏感度会更高。没有女孩子愿意顶着没有洗过的大油头出门逛街,如果必须要出门,也会选择戴顶帽子遮一下。微信安卓8。0。27最新官方正式版下载发布IT之家8月18日消息,微信安卓平台迎来了8。0。27正式版更新,本次更新安装包文件详细信息如下微信安卓正式版8。0。27(ARM64)安装包文件大小263。79MB更新时间202芯片价格暴跌,新能源汽车要跳楼价了?做梦众所周知,自开年以来,新能源汽车已经经历了几波涨价,其涨价幅度最高达一万块。而涨价的主要原因,一是动力电池原材料价格上涨,二是芯片价格暴涨。由于疫情等因素影响,缺芯问题日益凸显。供同是商标侵权,为何华为的商标比小米的更值钱?近日,华为技术有限公司(以下简称华为)诉深圳市尚派科技有限公司(以下简称尚派公司)侵犯商标权一案宣判,法院支持了华为提出的惩罚性赔偿诉讼请求,判令尚派公司赔偿华为500万元人民币。格力成立餐饮管理公司,要挑战外卖配送市场?格力成立餐饮管理公司,经营范围含外卖递送服务北京13名餐饮从业人员购买假冒健康证,已被警方拘留。详情请看红餐网每日餐讯。格力成立餐饮管理公司经营范围含外卖递送服务中国经济周刊消息,新疆主帅阿的江娶小5岁歌舞演员,女儿长相清纯成排球女神在篮球世界里,身高为王几乎成为一个真理。然而偶尔却会出现一个奇迹,小人物也能让大个子俯首贴耳。1。82米的阿的江,就是这样一个杰出的小人物。他从一名球员,最终成长为执掌八一队和新疆美澳科学家试图在实验室里复活灭绝物种袋狼,预计10年左右可成功袋狼(Thylacinuscynocephalus),也叫塔斯马尼亚虎(狼)被认为已经在地球上灭绝了,人们已知这种动物的最后一个活体,名字叫做本杰明,于1936年在澳大利亚霍巴特动进入中国25年后,阿迪达斯开始水土不服?作为风靡中国市场的国外运动品牌之一,阿迪达斯的CEO近日却承认自身在中国市场犯了错误,并一度登上微博热搜。大中华区销售额连续5个季度受挫,下调2022年的盈利目标阿迪达斯还能赢回中1台车没卖却融了50亿,这只造车独角兽为何受富爸爸们追捧?来源丨创业邦(IDichuangyebang)作者丨潘磊编辑丨子钺图源丨阿维塔官网新能源赛道又一只估值近百亿的独角兽诞生。近日,阿维塔完成超过25亿元人民币的A轮融资,投后估值近百如何配置保险最科学最省钱如何配置我们已经知道,保险保险,就是要在风险发生的时候,对冲风险的。哪里最需要保障呢,当然是发生风险时损失最大的地方,而家庭中谁发生风险时,家庭遭到的损失最大呢,显然是家庭支柱。因
早餐面食这样做,卷一卷,煎一煎,咸香劲道好消化,老少皆宜对于家长来说,做早餐,是一件非常难的事情,既要让家人吃得好,又要让他们吃得饱,关键是做出来的早餐还要很有营养,这可难倒了不少家长朋友啊,今天呢,我就来给大家分享一个比较简单的早餐做4月最该吃这种野菜,错过等一年!来源昆明日报掌上春城面条菜又叫甜甜菜麦瓶菜或野菠菜。面条菜含有丰富的维生素蛋白质胡萝卜素等营养成分。不仅可以作为蔬菜食用,同时也是种药用价值很高的药草,有健脾消肿清热解毒的功效,面来了来了今天狗做饭来了,姊妹们。昨天直播有人说想让狗做饭,今天让狗做饭。你这个不在水里头解冻,就这么往锅里下,我直接焯水不一样的吗?我以前给你做的时候从来都没摘过。今天做饭开心不?开心开心开心极了。公主鞋的养护建议正常穿搭建议无不良价值观引导公主鞋的养护建议公主鞋是一种高跟鞋,因为其造型精致具有华丽的装饰和时尚感,被广泛应用于各种正式场合。然而,由于其材质和做工较为精细,所以对于公主鞋的养护孔乙己的长衫,穿在谁的身上?学历不但是敲门砖,也是我下不来的高台,更是孔乙己脱不下的长衫。学历是敲门砖没错,但并不是我下不来的高台,或者说,我从没觉得那是什么高台,更别说什么脱不下的长衫。当代年轻人应该是迷茫阿斯顿马丁快闪魔都时尚新地标上周末,阿斯顿马丁为庆祝品牌110周年,在上海百年文化地标张园打造了一条阿斯顿马丁大道(ASTONMARTINAVENUE)。在售全系车型均亮相现场,全新F1AMR23赛车也首次出你的敷了真的有效果吗?教你四招,简单判断面膜的好坏!学会这四招,避免买到假面膜。假面膜,教你4招简单判断面膜的好坏。1看面膜包装执行标准号VDIILOTC(世),QBT2872(回误),才是符合生产标准的。2看防腐剂成分排名以下几种科普知识把春天吃进肚子前,先看看这份野菜食用指南!春天到了大家通过欣赏春天的风景享受春风的和煦来感受春天也可以通过味蕾来感受春天那么春天的味道又是什么样的呢?什么食物最能承载春天的味道?让我们一起来寻找春天的限定食物野菜把春天一口2道春天爱吃的家常炒菜,一素一荤,鲜香软糯味浓,吃得盘干碗净春天的蔬菜品种多,不管是淡雅素炒,还是浓香荤食,都能让人食欲大开,不舍放过每一口美味。今儿分享2道家常炒菜,一素一荤,做法简单,鲜味十足,软糯开胃,其丰富的营养不但能满足机体所需,鹤壁市委书记马富国到访豫信电科,推动在卫星产业等领域合作大河财立方消息4月10日上午,鹤壁市委书记马富国,市委副书记市政府市长赵宏宇等一行到访豫信电子科技集团(简称豫信电科),与豫信电科党委书记董事长李亚东等就信创产业发展数据要素市场化2023中超联赛第2轮武汉三镇vs北京国安开票了武汉三镇队将在2023赛季中超联赛第2轮中主场迎战北京国安,本场比赛球票线上发售通道已经开启,线下售票点将于4月17日1200开启。票务详情及购买方式请查看下图。三镇力量,武汉荣耀