Jamey's Jamey's
首页
导航站
  • 学习专栏

    • 《HTML》笔记
    • 《CSS》笔记
    • 《JavaScript》笔记
    • 《Vue》笔记
    • 《Git》笔记
    • 《规范》笔记
    • 《软技能》笔记
    • 《面试》笔记
    • 《持续集成&交付&部署》笔记
  • 踩坑专栏

    • 《Element-UI 实践系列》笔记
    • 《移动端 实践系列》笔记
    • 《综合》笔记
  • 配置专栏

    • 《环境系列》笔记
  • 极空间

    • Docker
  • 影视

    • movie
  • 编辑器笔记

    • 开发编辑器
  • 浏览器笔记

    • Chrome
  • Mac笔记

    • Mac
  • 跨界学习

    • 运营
  • 破解合集

    • 破解
  • 本站

    • 分类
    • 标签
    • 归档
  • 我的

    • 收藏
    • 书单
    • 关于

Jamey

首页
导航站
  • 学习专栏

    • 《HTML》笔记
    • 《CSS》笔记
    • 《JavaScript》笔记
    • 《Vue》笔记
    • 《Git》笔记
    • 《规范》笔记
    • 《软技能》笔记
    • 《面试》笔记
    • 《持续集成&交付&部署》笔记
  • 踩坑专栏

    • 《Element-UI 实践系列》笔记
    • 《移动端 实践系列》笔记
    • 《综合》笔记
  • 配置专栏

    • 《环境系列》笔记
  • 极空间

    • Docker
  • 影视

    • movie
  • 编辑器笔记

    • 开发编辑器
  • 浏览器笔记

    • Chrome
  • Mac笔记

    • Mac
  • 跨界学习

    • 运营
  • 破解合集

    • 破解
  • 本站

    • 分类
    • 标签
    • 归档
  • 我的

    • 收藏
    • 书单
    • 关于
  • 基础

  • 组件

  • 路由

  • 自定义指令

  • 实践踩坑

  • 知识点

    • vue 知识点总结
      • 📖. 前言
      • 一. 荣耀黄金
        • 1. Vue 的优点?Vue 的缺点?
        • 2. 为什么说 Vue 是一个渐进式框架?
        • 3. Vue 跟 React 的异同点?
        • 4. MVVM 是什么?和 MVC 有何区别呢?
        • 5. Vue 和 jQuery 的区别在哪?为什么放弃 jQuery 用 Vue?
        • 6. Vue的作者是谁?大声说出它的名字!!!
      • 二. 永恒钻石
        • 7. 为什么 data 是个函数并且返回一个对象呢?
        • 8. 使用过那些 Vue 的修饰符呢?
        • 9. 使用过哪些 Vue 的内部指令呢?
        • 10. 组件之间的传值方式有哪些?
        • 11. 路由有哪些模式呢?又有什么不同呢?
        • 12. 如何动态设置 class,动态 style?
        • 13. v-if 和 v-show 有何区别?
        • 14. computed 和 watch 有何区别?
        • 15. Vue 的生命周期,讲一讲?
        • 16. 为什么 v-if 和 v-for 不建议用在同一标签?
        • 17. vuex 有哪些属性?用处是什么?
      • 三. 至尊星耀
        • 18. 不需要响应式的数据应该怎么处理?
        • 19. watch 有哪些属性,分别有什么用?
        • 20. 父子组件生命周期顺序
        • 21. 对象新属性无法更新视图,删除属性无法更新视图,为什么?怎么办?
        • 22. 直接 arr[index] = xxx 无法更新视图怎么办?为什么?怎么办?
        • 23. 自定义指令
        • 24. 插槽的使用以及原理
        • 25. 为什么不建议用 index 做 key,为什么不建议用随机数做 key?
        • 26. 说说 nextTick 的用处?
        • 27. Vue 的 SSR 是什么?有什么好处?
      • 四. 最强王者
        • 28. Vue 响应式是怎么实现的?
        • 29. 为什么只对对象劫持,而对数组进行方法重写?
        • 30. Vue 的模板编译原理?
        • 31. Vue 的 computed 和 watch 的原理?
        • 32. Vue.set 方法的原理?
        • 33. Vue.delete 方法的原理?
        • 34. nextTick 的原理?
        • 35. key 有什么用?说说 diff 算法吧?
      • 五. 冷门知识
        • 36. 如果子组件改变 props 里的数据会发生什么
        • 37. props 怎么自定义验证
        • 38. watch 的 immediate 属性有什么用?
        • 39. watch 监听一个对象时,如何排除某些属性的监听
        • 40. 审查元素时发现 data-v-xxx,这是啥?
        • 41. computed 如何实现传参?
        • 42. vue 的 hook 的使用
        • 43. provide 和 inject 时响应式的吗?
        • 44. Vue 的 el 属性和 $mount 优先级?
        • 45. 动态指令和参数使用过吗?
        • 46. 相同的路由组件如何重新渲染?
        • 47. 自定义 v-model
        • 48. 如何将获取 data 中某一个数据的初始状态?
        • 49. 为什么不建议 v-for 和 v-if 同时存在
        • 50. 计算变量时,methods 和 computed 哪个好?
  • 《Vue》笔记
  • 知识点
Jamey
2022-07-25
目录

vue 知识点总结

# vue 知识点总结

# 📖. 前言

工作多年总结的 vue 知识点。

# 一. 荣耀黄金

# 1. Vue 的优点?Vue 的缺点?

优点:渐进式,组件化,轻量级,虚拟dom,响应式,单页面路由,数据与视图分开。

缺点:单页面不利于 seo,不支持 IE8 一下,首屏加载时间长。

# 2. 为什么说 Vue 是一个渐进式框架?

渐进式:通俗点讲就是,你想用啥你就用啥,咱也不强求你。你想用 component 就用,不用也行,你想用 vuex 就用,不用也可以。

vue_07-22_01

# 3. Vue 跟 React 的异同点?

相同点:

  • 都用了虚拟 dom
  • 组件化开发
  • 都是单项数据流(父子组件之间,不建议子修改父传下来的数据)
  • 都支持服务端渲染

不同点:

  • React 的 JSX,Vue 的 template
  • 数据变化,React 手动(setState),Vue 自动(初始化已响应式处理,Object.defineProperty)
  • React 单向帮顶,Vue 双向绑定
  • React 的 Redux,Vue 的 Vuex

# 4. MVVM 是什么?和 MVC 有何区别呢?

MVC

  • Model(模型):负责从数据库中取数据
  • View(视图):负责展示数据的地方
  • Controller(控制器):用户交互的地方,例如点击事件等等
  • 思想:Controller 将 Model 的数据展示在 View 上

vue_07-22_02

MVVM

  • VM:也就是 View-Model,做了两件事达到了数据的双向绑定。一是将【模型】转换成【视图】,即将后端传递的数据转换成所看到的页面。实现的方式是:数据绑定。 二是将【视图】转换成【模型】,即将所看到的页面转换成后端的数据。实现的方式是:DOM 事件监听。
  • 思想:实现了 Vie 和 Model 的自动同步,也就是当 Model 的属性改变时,我们不用再自己手动操作 Dom 元素,来改变 View 的显示,而是改变属性后该属性对应 View 层显示会自动改变(对应 Vue 数据驱动的思想)

vue_07-22_03

区别

整体来看,MVVM 比 MVC 精简很多,不仅简化了业务与界面的依赖,还解决了数据频繁更新的问题,不用再用选择器操作 DOM 元素。因为在 MVVM 中,View 不知道 Model 的存在,Model 和 ViewModel 也观察不到 View, 这种低耦合模式提高代码的可重用性。

Vue 是不是 MVVM 框架?

Vue 是 MVVM 框架,但是不是严格符合 MVVM,因为 MVVM 规定 Model 和 View 不能直接通信,而 Vue 的 ref 可以做到这点。

# 5. Vue 和 jQuery 的区别在哪?为什么放弃 jQuery 用 Vue?

  1. jQuery 是直接操作 DOM,Vue 不直接操作 DOM,Vue 的数据与视图是分开的,Vue 只需要操作数据即可。
  2. 在操作 DOM 频繁的场景里,jQuery 的操作 DOM 行为是频繁的,而 Vue 利用虚拟 DOM 的技术,大大提高了更新 DOM 时的性能。
  3. Vue 中不倡导直接操作 DOM,开发者只需要把大部分精力放在数据层面上。
  4. Vue 集成的一些库 ,大大提高开发效率,比如 Vuex,Router 等。

# 6. Vue的作者是谁?大声说出它的名字!!!

他的名字就是:鱿鱼西

# 二. 永恒钻石

# 7. 为什么 data 是个函数并且返回一个对象呢?

data 之所以是一个函数,是因为一个组件可能会多处调用,而每一次调用就会执行 data函数 并返回新的数据对象,这样,可以避免多处调用之间的 数据污染。

# 8. 使用过那些 Vue 的修饰符呢?

vue_07-22_04

# 9. 使用过哪些 Vue 的内部指令呢?

vue_07-22_05

# 10. 组件之间的传值方式有哪些?

  • 父组件传值给子组件,子组件使用 props 进行接收。
  • 子组件传值给父组件,子组件使用 $emit + 事件 对父组件进行传值。
  • 组件中可以使用 $parenet 和 $children 获取到父组件实例和子组件实例,进而获取数据。
  • 使用 $attr 和 $listeners,在对一些组件进行二次封装的时候可以方便传值,例如 A → B → C
  • 使用 $refs 获取组件实例,进而获取数据。
  • 使用 Vuex 进行状态管理。
  • 使用 eventBus 进行跨组件触发事件,进而传递数据。
  • 使用 provide 和 inject,官方建议我们不要用这个,我在看 ElementUI 源码时发现大量使用。
  • 使用浏览器本地缓存,例如 localStorage

# 11. 路由有哪些模式呢?又有什么不同呢?

  • hash模式:通过 #号 后面的内容梗概,触发 hashchange 事件,实现路由切换。
  • history模式:通过 pushState 和 replaceState 切换 url,实现路由切换,需要后端配合。

# 12. 如何动态设置 class,动态 style?

  • 动态 class 对象:<div :class="{ 'is-active': true, 'red': isRed }"></div>
  • 动态 class 数组:<div :class="['is-active', isRed ? 'red' : '' ]"></div>
  • 动态 style 对象:<div :style="{ color: textColor, fontSize: '18px' }"></div>
  • 动态 style 数组:<div :style="[{ color: textColor, fontSize: '18px' }, { fontWeight: '300' }]"></div>

# 13. v-if 和 v-show 有何区别?

    1. v-if 是通过控制 dom 元素的删除和生成来实现显隐,每一次显隐都会使组件重新跑一遍生命周期,因为显隐决定了组件的生成和销毁
    1. v-show 是通过控制 dom 元素的 css 样式来实现显隐,不会销毁
    1. 频繁或者大数量显隐使用 v-show,否则使用 v-if

# 14. computed 和 watch 有何区别?

    1. computed 是依赖已有的变量来计算一个目标变量,大多数情况都是多个变量凑在一起计算出一个变量,并且 computed 具有缓存机制,依赖值不变的情况下其会直接读取缓存进行复用,computed 不能进行异步操作。
    1. watch 是监听某一个变量的变化,并执行相应的回调函数,通常是一个变量的变化决定多个变量的变化,watch 可以进行异步操作。
    1. 简单记就是:一般情况下 computed 是多对一,watch 是一对多

# 15. Vue 的生命周期,讲一讲?

vue_07-22_06

# 16. 为什么 v-if 和 v-for 不建议用在同一标签?

在 Vue2 中,v-for 优先级是高于 v-if 的,咱们来看例子。

<div v-for="item in [1, 2, 3, 4, 5, 6, 7]" v-if="item !== 3">
  {{item}}
</div>
1
2
3

上面的写法是 v-for 和 v-if 同时存在,会先把 7 个元素都遍历出来,然后再一个个判断是否为 3,并把 3 给隐藏掉,这样的坏处就是,渲染了无用的 3 节点,增加无用的 dom 操作,建议使用 computed 来解决这个问题:

<template>
  <div v-for="item in list">
      {{ item }}
  </div>
</template>

<script >
  export default {
    computed() {
      list() {
        return [1, 2, 3, 4, 5, 6, 7].filter(item => item !== 3)
      }
    }
  }
</script>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15

# 17. vuex 有哪些属性?用处是什么?

vue_07-22_07

  • State:定义了应用状态的数据结构,可以在这里设置默认的初始状态。
  • Getter:允许组件从 Store 中获取数据,mapGetters 辅助函数仅仅是将 store 中的 getter 映射到局部计算属性。
  • Mutation:是唯一更改 store 中状态的方法,且必须是同步函数。
  • Action:用于提交 mutation,而不是直接变更状态,可以包含任意异步操作。
  • Module:允许将单一的 Store 拆分为多个 store 且同时保存在单一的状态树中。

# 三. 至尊星耀

# 18. 不需要响应式的数据应该怎么处理?

在我们的 Vue 开发中,会有一些数据,从始至终都 未曾改变过,这种 死数据,既然 不改变,那也就 不需要对它做响应式处理 了,不然只会做一些无用功消耗性能, 比如一些写死的下拉框,写死的表格数据,这些数据量大的 死数据,如果都进行响应式处理,那会消耗大量性能。

// 方法一:将数据定义在data之外
data () {
  this.list1 = { xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx }
  this.list2 = { xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx }
  this.list3 = { xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx }
  this.list4 = { xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx }
  this.list5 = { xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx }
  return {}
}

// 方法二:Object.freeze()
data () {
  return {
    list1: Object.freeze({xxxxxxxxxxxxxxxxxxxxxxxx}),
    list2: Object.freeze({xxxxxxxxxxxxxxxxxxxxxxxx}),
    list3: Object.freeze({xxxxxxxxxxxxxxxxxxxxxxxx}),
    list4: Object.freeze({xxxxxxxxxxxxxxxxxxxxxxxx}),
    list5: Object.freeze({xxxxxxxxxxxxxxxxxxxxxxxx}),
  }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20

# 19. watch 有哪些属性,分别有什么用?

当我们监听一个基本数据类型时:

watch: {
  value () {
    // do something
  }
}
1
2
3
4
5

当我们监听一个引用数据类型时:

watch: {
  obj: {
    handler() { // 执行回调
      // do something
    },
    deep: true, // 是否进行深度监听
    immediate: true // 是否初始执行 handler 函数
  }
}
1
2
3
4
5
6
7
8
9

# 20. 父子组件生命周期顺序

父 beforeCreate → 父created → 父beforeMount → 子beforeCreate → 子created → 子beforeMount → 子mounted → 父mounted

# 21. 对象新属性无法更新视图,删除属性无法更新视图,为什么?怎么办?

  • 原因:Object.defineProperty 没有对对象的新属性进行属性劫持
  • 对象新属性无法更新视图:使用 Vue.$set(obj, key, value),组件中 this.$set(obj, key, value)
  • 删除属性无法更新视图:使用 Vue.$delete(obj, key),组件中 this.$delete(obj, key)

# 22. 直接 arr[index] = xxx 无法更新视图怎么办?为什么?怎么办?

  • 原因:Vue 没有对数组进行 Object.defineProperty 的属性劫持,所以直接 arr[index] = xxx 是无法更新视图的
  • 使用数组的 splice 方法,arr.splice(index, 1, item)
  • 使用 Vue.$set(arr, index, value)

# 23. 自定义指令

# 24. 插槽的使用以及原理

# 25. 为什么不建议用 index 做 key,为什么不建议用随机数做 key?

举个例子:

<div v-for="(item, index) in list" :key="index">{{ item.name }}</div>

list: [
  { name: '小明', id: '123' },
  { name: '小红', id: '124' },
  { name: '小花', id: '125' }
]

渲染为
<div key="0">小明</div>
<div key="1">小红</div>
<div key="2">小花</div>

现在我执行 list.unshift({ name: '小林', id: '122' })

渲染为
<div key="0">小林</div>
<div key="1">小明</div>
<div key="2">小红</div>
<div key="3">小花</div>

新旧对比
<div key="0">小明</div>  <div key="0">小林</div>
<div key="1">小红</div>  <div key="1">小明</div>
<div key="2">小花</div>  <div key="2">小红</div>
<div key="3">小花</div>

可以看出,如果用 index 做 key 的话,其实是更新了原有的三项,并新增了小花,虽然达到了渲染目的,但是损耗性能

现在我们使用 id 来做 key,渲染为
<div key="123">小明</div>
<div key="124">小红</div>
<div key="125">小花</div>

现在我执行 list.unshift({ name: '小林', id: '122' }),渲染为

<div key="122">小林</div>
<div key="123">小明</div>
<div key="124">小红</div>
<div key="125">小花</div>

新旧对比
                          <div key="122">小林</div>
<div key="123">小明</div>  <div key="123">小明</div>
<div key="124">小红</div>  <div key="124">小红</div>
<div key="125">小花</div>  <div key="125">小花</div>

可以看出,原有的三项都不变,只是新增了小林这个人,这才是最理想的结果
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48

用 index 和用 随机数 都是同理,随机数 每次都在变,做不到专一性,很 渣男,也很消耗性能,所以,拒绝 渣男,选择 老实人。

# 26. 说说 nextTick 的用处?

我举个例子,在vue中:

this.name = '林三心';
this.age = 18;
this.gender = '男';
1
2
3

我们修改了三个变量,那问题来了,是每修改一次,DOM就更新一次吗?不是的,Vue 采用的是 异步更新 的策略,通俗点说就是,同一事件循环内 多次修改,会 统一 进行一次 视图更新,这样才能节省性能嘛。

懂了上面,那你应该也看得懂下面的例子了吧:

<div ref="testDiv">{{ name }}</div>

name: '小林'

this.name = '林三心'
console.log(this.$refs.testDiv.innerHTML) // 这里是啥呢
1
2
3
4
5
6

答案是 “小林”,前面说了,Vue是 异步更新,所以数据一更新,视图却还没更新,所以拿到的还是上一次的旧视图数据,那么想要拿到最新视图数据怎么办呢?

this.name = '林三心'
this.$nextTick(() => {
  console.log(this.$refs.testDiv.innerHTML); // 林三心
})
1
2
3
4

# 27. Vue 的 SSR 是什么?有什么好处?

  • SSR 就是服务端渲染。
  • 基于 nodejs serve 服务环境开发,所有 html 代码在服务端渲染。
  • 数据返回给前端,然后前段进行 “激活”,即可成为浏览器识别的 html 代码。
  • SSR 首次加载更快,有更好的用户体验,有更好的 seo 优化,因为爬虫能看到整个页面的内容,如果是 vue 项目,由于数据还要经过解析,这就造成爬虫并不会等待你的数据加载完成,所以 Vue 项目的 seo 体验并不是很好。

# 四. 最强王者

# 28. Vue 响应式是怎么实现的?

整体思路是 数据劫持 + 观察者模式。

对象内部通过 defineReactive 方法,使用 Object.defineProperty 将属性进行劫持(只会劫持已经存在的属性),数组则是通过重写数组方法来实现。当页面使用对应属性时,每个属性都拥有自己的 dep 属性, 存放它所以来的 watcher(依赖收集),当属性变化后会通知自己对应的 watcher 去更新(派发更新)。

const { arrayMethods } = require('./array')

class Observer {
  constructor(value) {
    Object.defineProperty(value, '__ob__', {
      value: this,
      enumerable: false,
      writable: true,
      configurable: true
    })
    if(Array.isArray(value)) {
      value.__proto__ = arrayMethods
      this.observeArray(value)
    } else {
      this.walk(value)
    }
  }

  walk(data) {
    let keys = Object.keys(data)
    for(let i = 0; i < keys.length; i++) {
      const key = keys[i]
      const value = data[key]
      defineReactive(data, key, value)
    }
  }
  
  observeArray(items) {
    for(let i = 0; i < items.length; i++) {
      observe(items[i])
    }
  }
}

function defineReactive(data, key, value) {
  const childOb = observe(value)
  
  const dep = new Dep()
  
  Object.defineProperty(data, key, {
    get() {
      console.log('获取值')
      if (Dep.target) {
        dep.depend()

        if (childOb) {
          childOb.dep.depend()

          if (Array.isArray(value)) {
            dependArray(value)
          }
        }
      }
      return value
    },
    set(newVal) {
      if (newVal === value) return
      observe(newVal)
      value = newVal
      dep.notify()
    }
  })
}

function observe(value) {
  if (Object.prototype.toString.call(value) === '[object Object]' || Array.isArray(value)) {
    return new Observer(value)
  }
}

function dependArray(value) {
  for(let e, i = 0, l = value.length; i < l; i++) {
    e = value[i]
  
    e && e.__ob__ && e.__ob__.dep.depend()
  
    if (Array.isArray(e)) {
      dependArray(e)
    }
  }
}

// array.js
const arrayProto = Array.prototype

const arrayMethods = Object.create(arrayProto)

const methodsToPatch = [
  'push',
  'pop',
  'shift',
  'unshift',
  'splice',
  'reverse',
  'sort'
]

methodsToPatch.forEach(method => {
  arrayMethods[method] = function (...args) {
    const result = arrayProto[method].apply(this, args)
  
    const ob = this.__ob__
  
    var inserted
  
    switch (method) {
      case 'push':
      case 'unshift':
        inserted = args
        break;
      case 'splice':
        inserted = args.slice(2)
      default:
        break;
    }
  
    if (inserted) ob.observeArray(inserted)
  
    ob.dep.notify()
  
    return result
  }
})
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123

# 29. 为什么只对对象劫持,而对数组进行方法重写?

因为对象最多也就几十个属性,拦截起来数量不多,但是数组可能会有几百几千项,拦截起来非常耗性能,所以直接重写数组原型上的方法,是比较节省性能的方案。

# 30. Vue 的模板编译原理?

# 31. Vue 的 computed 和 watch 的原理?

# 32. Vue.set 方法的原理?

function set(target, key, val) {
  // 判断是否是数组
  if (Array.isArray(target)) {
    // 判断谁大谁小
    target.length = Math.max(target.length, key)
    // 执行splice
    target.splice(key, 1, val)
    return val
  }
  
  const ob = target.__ob__
  
  // 如果此对象没有不是响应式对象,直接设置并返回
  if (key in target && !(key in target.prototype) || !ob) {
    target[key] = val
    return val
  }
  
  // 否则,新增属性,并响应式处理
  defineReactive(target, key, val)
  return val
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22

# 33. Vue.delete 方法的原理?

function del (target, key) {
  // 判断是否为数组
  if (Array.isArray(target)) {
    // 执行splice
    target.splice(key, 1)
    return
  }
  
  const ob = target.__ob__
  
  // 对象本身就没有这个属性,直接返回
  if (!(key in target)) return
  
  
  // 否则,删除这个属性
  delete target[key]
  
  // 判断是否是响应式对象,不是的话,直接返回
  if (!ob) return
  // 是的话,删除后要通知视图更新
  ob.dep.notify()
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22

# 34. nextTick 的原理?

let callbacks = []; //回调函数
let pending = false;
function flushCallbacks() {
  pending = false; //把标志还原为false
  // 依次执行回调
  for (let i = 0; i < callbacks.length; i++) {
    callbacks[i]();
  }
}
let timerFunc; //先采用微任务并按照优先级优雅降级的方式实现异步刷新
if (typeof Promise !== "undefined") {
  // 如果支持promise
  const p = Promise.resolve();
  timerFunc = () => {
    p.then(flushCallbacks);
  };
} else if (typeof MutationObserver !== "undefined") {
  // MutationObserver 主要是监听dom变化 也是一个异步方法
  let counter = 1;
  const observer = new MutationObserver(flushCallbacks);
  const textNode = document.createTextNode(String(counter));
  observer.observe(textNode, {
    characterData: true,
  });
  timerFunc = () => {
    counter = (counter + 1) % 2;
    textNode.data = String(counter);
  };
} else if (typeof setImmediate !== "undefined") {
  // 如果前面都不支持 判断setImmediate
  timerFunc = () => {
    setImmediate(flushCallbacks);
  };
} else {
  // 最后降级采用setTimeout
  timerFunc = () => {
    setTimeout(flushCallbacks, 0);
  };
}

export function nextTick(cb) {
  callbacks.push(cb);
  if (!pending) {
    pending = true;
    timerFunc();
  }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47

# 35. key 有什么用?说说 diff 算法吧?

直接看这篇吧:为什么 Vue 中不要用 index 作为 key?(diff 算法详解) (opens new window)

我讲的没他好

# 五. 冷门知识

# 36. 如果子组件改变 props 里的数据会发生什么

  • 改变的 props 数据是基本类型

如果修改的是基本类型,则会报错

props: {
  num: Number,
  }
created() {
  this.num = 999
}
1
2
3
4
5
6

报错

[Vue warn]: Avoid mutating a prop directiy since the value will be overwritten whenever the parent component re-renders. Instead, use a data or computed property based on the prop's value. Prop being mutated; "index"

found in

  • 改变的 props 数据是引用类型
props: {
  item: {
    default: () => ({}),
  }
}

created() {
    // 不报错,并且父级数据会跟着变
    this.item.name = 'sanxin';
    
    // 会报错,跟基础类型报错一样
    this.item = 'sss'
  },
1
2
3
4
5
6
7
8
9
10
11
12
13

# 37. props 怎么自定义验证

props: {
  num: {
    default: 1,
    validator: function (value) {
      // 返回值为false则验证不通过,报错
      return [
        1, 2, 3, 4, 5
      ].indexOf(value) !== -1
    }
  }
}
1
2
3
4
5
6
7
8
9
10
11

# 38. watch 的 immediate 属性有什么用?

比如平时 created 时要请求一次数据,并且当搜索值改变,也要请求数据,我们会这么写:

created(){
  this.getList()
},

watch: {
  searchInputValue(){
    this.getList()
  }
}
1
2
3
4
5
6
7
8
9

使用 immediate 完全可以这么写,当它为 true 时,会初始执行一次

watch: {
  searchInputValue:{
    handler: 'getList',
      immediate: true
  }
}
1
2
3
4
5
6

# 39. watch 监听一个对象时,如何排除某些属性的监听

下面代码是,params 发生改变就重新请求数据,无论是 a,b,c,d 属性改变

data() {
  return {
    params: {
      a: 1,
      b: 2,
      c: 3,
      d: 4
    },
  };
},

watch: {
  params: {
    deep: true,
    handler() {
      this.getList;
    },
  },
  }
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19

但是如果我只想要 a,b 改变时重新请求,c,d 改变时不重新请求呢

mounted() {
  Object.keys(this.params)
    .filter((_) => !["c", "d"].includes(_)) // 排除对c,d属性的监听
    .forEach((_) => {
      this.$watch((vm) => vm.params[_], handler, {
        deep: true,
      });
    });
},

data() {
  return {
    params: {
      a: 1,
      b: 2,
      c: 3,
      d: 4
    },
  };
},

watch: {
  params: {
    deep: true,
    handler() {
      this.getList;
    },
  },
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29

# 40. 审查元素时发现 data-v-xxx,这是啥?

vue_07-22_08

这是在标记 vue 文件中 css 时使用 scoped 标记产生的,因为要保证各个文件中的 css 不相互影响,给每个 component 都做了唯一的标记,所以每引入一个 component 就会出现一个新的 data-v-xxx 标记。

# 41. computed 如何实现传参?

// html
<div>{{ total(3) }}

// js
computed: {
  total() {
    return function(n) {
      return n * this.num
    }
  },
}
1
2
3
4
5
6
7
8
9
10
11

# 42. vue 的 hook 的使用

  • 同一组件中使用

这是我们常用的使用定时器的方式

export default{
  data(){
    timer:null  
  },
  
  mounted(){
    this.timer = setInterval(()=>{
      //具体执行内容
      console.log('1');
    },1000);
  },
  
  beforeDestory(){
    clearInterval(this.timer);
    this.timer = null;
  }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17

上面的做法不好的地方在于:得全局多定义一个 timer 变量,可以使用 hook 这么做:

export default{
  methods:{
    fn(){
      let timer = setInterval(()=>{
        //具体执行代码
        console.log('1');
      },1000);
      this.$once('hook:beforeDestroy',()=>{
        clearInterval(timer);
        timer = null;
      })
    }
  }
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
  • 父子组件使用

如果子组件需要再 mounted 时触发父组件的某一个函数,平时都会这么写:

//父组件
<rl-child @childMounted="childMountedHandle"/>

method () {
  childMountedHandle() {
    // do something...
  }
},

// 子组件
mounted () {
  this.$emit('childMounted');
},
1
2
3
4
5
6
7
8
9
10
11
12
13

使用 hook 的话可以更方便:

//父组件
<rl-child @hook:mounted="childMountedHandle"/>

method () {
  childMountedHandle() {
  // do something...
  }
},
1
2
3
4
5
6
7
8

# 43. provide 和 inject 时响应式的吗?

// 祖先组件
provide(){
  return {
    // keyName: { name: this.name }, // value 是对象才能实现响应式,也就是引用类型
    keyName: this.changeValue // 通过函数的方式也可以[注意,这里是把函数作为 value,而不是 this.changeValue()]
    // keyName: 'test' value 如果是基本类型,就无法实现响应式
  }
},

data(){
  return {
	  name:'张三'
  }
},

methods: {
  changeValue(){
    this.name = '改变后的名字-李四'
  }
}  
  
// 后代组件
inject:['keyName'],

create(){
  console.log(this.keyName) // 改变后的名字 - 李四
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27

# 44. Vue 的 el 属性和 $mount 优先级?

比如下面这种情况,Vue 会渲染到哪个节点上?

new Vue({
  router,
  store,
  el: '#app',
  render: h => h(App)
}).$mount('#ggg')
1
2
3
4
5
6

这是官方的一张图,可以看出 el 和 $mount 同时存在时,el 优先级 > $mount

vue_07-22_09

# 45. 动态指令和参数使用过吗?

<template>
  ...
  <aButton @[someEvent]="handleSomeEvent()" :[someProps]="1000" />...
</template>
<script>
  ...
  data(){
    return{
      ...
      someEvent: someCondition ? "click" : "dbclick",
      someProps: someCondition ? "num" : "price"
    }
  },
  methods: {
    handleSomeEvent(){
      // handle some event
    }
  }  
</script>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19

# 46. 相同的路由组件如何重新渲染?

开发人员经常遇到的情况是,多个路由解析为同一个 Vue 组件。问题是,Vue出于性能原因,默认情况下共享组件将不会重新渲染,如果你尝试在使用相同组件的路由之间进行切换,则不会发生任何变化。

const routes = [
  {
    path: "/a",
    component: MyComponent
  },
  {
    path: "/b",
    component: MyComponent
  },
];
1
2
3
4
5
6
7
8
9
10

如果依然想重新渲染,怎么办呢?可以使用 key

<template>
  <router-view :key="$route.path"></router-view>
</template>
1
2
3

# 47. 自定义 v-model

默认情况下,v-model 是 @input 事件侦听器和 :value 属性上的语法糖。但是,你可以在你的 Vue 组件中指定一个模型属性来定义使用什么事件和 value 属性 —— 非常棒!

export default: {
  model: {
    event: 'change',
    prop: 'checked'  
  }
}
1
2
3
4
5
6

# 48. 如何将获取 data 中某一个数据的初始状态?

在开发中,有时候需要拿初始状态去计算。例如

<script>
export default {
  data() {
    return {
      num: 10
  },
      
  mounted() {
    this.num = 1000
  },
    
  methods: {
    howMuch() {
      // 计算出 num 增加了多少,那就是 1000 - 初始值
      // 可以通过 this.$options.data().xxx 来获取初始值
      console.log(1000 - this.$options.data().num)
    }
  }
}
</script>
    

  

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24

# 49. 为什么不建议 v-for 和 v-if 同时存在

<div v-for="item in [1, 2, 3, 4, 5, 6, 7]" v-if="item !== 3">
  {{ item }}
</div>
1
2
3

上面的写法是 v-for 和 v-if 同时存在,会先把 7 个元素都遍历出来,然后再一个个判断是否为 3,并把 3 给隐藏掉,这样的坏处就是,渲染了无用的 3 节点,增加无用的 dom 操作,建议使用 computed 来解决这个问题:

<template>
  <div v-for="item in list">
    {{ item }}
  </div>
</template>

<script>
export default {
  computed() {
    list()
    {
      return [1, 2, 3, 4, 5, 6, 7].filter(item => item !== 3)
    }
  }
}
</script>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16

# 50. 计算变量时,methods 和 computed 哪个好?

<template>
  <div>
    <div>{{ howMuch1() }}</div>
    <div>{{ howMuch2 }}</div>
    <div>{{ index }}</div>
  </div>
</template>

<script>
  export default {
    data () {
      return {
        index: 0
      }
    },

    computed: {
      howMuch2() {
        return this.num + this.price
      }
    },

    methods: {
      howMuch1() {
        return this.num + this.price
      }
    }
  }
</script>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29

computed 会好一些,因为 computed 会有 缓存。例如 index 由 0 变成 1,那么会触发视图更新,这时候 methods 会重新执行一次,而 computed 不会,因为 computed 依赖的两个变量 num 和 price 都没变。

#Vue.js2
上次更新: 2022/08/09, 10:43:12
vue-cli4 配置打包命令

← vue-cli4 配置打包命令

Theme by Vdoing | Copyright © 2017-2023 Jamey | blog 闽ICP备19022664号
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式