• vue2.深入响应式原理


    vue2.x是利用Object.defineProperty劫持对象或对象的属性(指的是在访问或者修改对象的某个属性时,通过一段代码拦截这个行为,进行额外的操作或者修改返回结果)的访问器,在属性值发生变化时获取属性值变化, 从而进行后续操作。

    数据代理的的demo(通过一个对象代理对另一个对象属性的操作)

    1. var a = {age:20}
    2. var b = {height:30}
    3. Object.defineProperty(b,'age',{
    4. get(){
    5. return a.age
    6. },
    7. set(newvalue){
    8. a.age = newvalue
    9. }
    10. })

    1、Object.defineProperty在js中的描述:

    Object.defineProperty(obj, prop, descriptor) 直接在一个对象上定义一个属性,或者修改一个对象的现有 属性,并返回这个对象。

    参数:obj 要在其上定义属性的对象;prop 要定义或修改的属性的名称;descriptor 将被定义或修改的属性描述符。

    返回值: 传递给函数的对象obj
     

    1. html>
    2. <html lang="en">
    3. <head>
    4. <meta charset="UTF-8">
    5. <meta name="viewport" content="width=device-width, initial-scale=1.0">
    6. <title>数据响应式原理title>
    7. head>
    8. <body>
    9. <div id="app">
    10. Hello Vue
    11. div>
    12. <script>
    13. //模拟Vue实例中的data选项
    14. let data={
    15. msg:'Hello Vue'
    16. }
    17. //模拟Vue的实例
    18. let vm={};
    19. //数据劫持,当访问或者设置vm中的成员的时候,做一些干预操作
    20. Object.defineProperty(vm,'msg',{
    21. //可枚举(即可被遍历)
    22. enumerable:true,
    23. //可配置(可以使用delete删除,可以通过defineProperty重新定义)
    24. configurable:true,
    25. //当获取值时执行
    26. get(){
    27. console.log('getter:',data.msg);
    28. return data.msg;
    29. },
    30. //当设置、更新msg变量时执行
    31. set(newValue){
    32. console.log("setter:",newValue);
    33. if(data.msg===newValue){
    34. return;//前后数据相同,则不用做操作DOM的多余操作
    35. }
    36. data.msg=newValue;
    37. document.querySelector("#app").textContent=newValue;
    38. }
    39. })
    40. //测试setter
    41. vm.msg="Hello 响应式原理";
    42. //测试getter
    43. console.log(vm.msg);
    44. script>
    45. body>
    46. html>

    上面实例只是对msg这个属性实现了响应式,那Vue中data选项有多个属性,怎么做到让它们都成为响应式呢

    1. html>
    2. <html lang="en">
    3. <head>
    4. <meta charset="UTF-8">
    5. <meta name="viewport" content="width=device-width, initial-scale=1.0">
    6. <title>数据响应式原理title>
    7. head>
    8. <body>
    9. <div id="app">
    10. Hello Vue
    11. div>
    12. <script>
    13. //模拟Vue实例中的data选项
    14. let data={
    15. msg:'Hello Vue',
    16. count:0
    17. }
    18. //模拟Vue的实例
    19. let vm={};
    20. function defineProperties(data){
    21. //循环给每个属性使用Object.defineProperty()
    22. Object.keys(data).forEach(key => {
    23. //数据劫持,当访问或者设置vm中的成员的时候,做一些干预操作
    24. Object.defineProperty(vm,key,{
    25. //可枚举(即可被遍历)
    26. enumerable:true,
    27. //可配置(可以使用delete删除,可以通过defineProperty重新定义)
    28. configurable:true,
    29. //当获取值时执行
    30. get(){
    31. console.log('getter:',data[key]);
    32. return data[key];
    33. },
    34. //当设置、更新msg变量时执行
    35. set(newValue){
    36. console.log("setter:",newValue);
    37. if(data[key]===newValue){
    38. return;//前后数据相同,则不用做操作DOM的多余操作
    39. }
    40. data[key]=newValue;
    41. document.querySelector("#app").textContent=newValue;
    42. }
    43. })
    44. });
    45. }
    46. //执行该函数,使每个属性添加响应式
    47. defineProperties(data);
    48. //测试setter
    49. vm.msg="Hello 响应式原理";
    50. //测试getter
    51. console.log(vm.msg);
    52. script>
    53. body>
    54. html>

    2、基于Object.defineProperty的数据劫持优势以及实现方式

    Object.defineProperty的对象以及对象属性的劫持有以下优势:

    (1)无需显式调用,如Vue2.x使用Object.defineProperty对象以及对象属性的劫持+发布订阅模式,只要数据发生变化直接通知变化 并驱动视图更新。

    (2)可在set函数中精确得知变化数据而不用逐个遍历属性获取变化值,减少性能损耗。

    实现思路:

    (1)利用Object.defineProperty重新定义一遍目标对象,完成对目标对象的劫持,在属性值变化后即触发set方法 后通知订阅者,告诉该对象的某个属性值发生了变化。

    (2)解析器Compile解析模板中的指令,收集指令所依赖的方法和数据,等待数据变化然后进行渲染。

    (3)Watcher在收到属性值发生变化后,根据解析器Compile提供的指令进行视图渲染。

    检测变化的注意事项
    Object.defineProperty() 可以监测到属性的获取、修改,但是新增、删除监测不到,

    1. html>
    2. <html>
    3. <head>
    4. <meta charset="utf-8">
    5. <title>title>
    6. head>
    7. <body>
    8. <div id="app">
    9. <h2 @click="qq">{{isAgree.msg}}h2>
    10. <h2>{{isAgree.ww}}h2>
    11. <h2 v-for="(item,index) in isAgree.web" :key="index">{{item}}h2>
    12. <button @click="updateweb">点击button>
    13. div>
    14. <script src="https://cdn.jsdelivr.net/npm/vue@2.6.10/dist/vue.js">script>
    15. <script>
    16. const app = new Vue({
    17. el: '#app',
    18. data: {
    19. isAgree: {
    20. msg: 'zhangsan',
    21. web:['html','js']
    22. }
    23. },
    24. methods: {
    25. qq() {
    26. /* 直接添加无效
    27. 必须使用Vue.set或者this.$set */
    28. /* this.isAgree.ww = "lisi" */
    29. /* Vue.set(this.isAgree,'ww','lisi') */
    30. /* this.$set(this.isAgree,'qq','2') */
    31. /* 使用delete不可用 */
    32. /* console.log(this.isAgree.msg);
    33. delete this.isAgree.msg
    34. console.log(this.isAgree.msg); */
    35. /* 可以使用Vue.delete或者this.$delete
    36. Vue.delete(this.isAgree, 'msg') */
    37. /* this.$delete(this.isAgree, 'msg') */
    38. },
    39. updateweb(){
    40. /* this.isAgree.web[0]="vue" 无效*/
    41. /* this.$set(this.isAgree.web,0,'vue') 有效*/
    42. /* this.isAgree.web.splice(0,1,'vue') 有效*/
    43. }
    44. }
    45. })
    46. script>
    47. body>
    48. html>

    由于 JavaScript 的限制,Vue 不能检测数组和对象的变化。尽管如此我们还是有一些办法来回避这些限制并保证它们的响应性。

    Vue 无法检测 property 的添加或移除。由于 Vue 会在初始化实例时对 property 执行 getter/setter 转化,所以 property 必须在 data 对象上存在才能让 Vue 将它转换为响应式的

    1. var vm = new Vue({
    2. data:{
    3. a:1
    4. }
    5. })
    6. // `vm.a` 是响应式的
    7. vm.b = 2
    8. // `vm.b` 是非响应式的

    对于已经创建的实例,Vue 不允许动态添加根级别的响应式 property。但是,可以使用 Vue.set(object, propertyName, value) 方法向嵌套对象添加响应式 property。例如,对于:

    Vue.set(vm.someObject, 'b', 2)


    您还可以使用 vm.$set 实例方法,这也是全局 Vue.set 方法的别名:

    this.$set(this.someObject,'b',2)


    有时你可能需要为已有对象赋值多个新 property,比如使用 Object.assign() 或 _.extend()。但是,这样添加到对象上的新 property 不会触发更新。在这种情况下,你应该用原对象与要混合进去的对象的 property 一起创建一个新的对象。

    1. // 代替 `Object.assign(this.someObject, { a: 1, b: 2 })`
    2. this.someObject = Object.assign({}, this.someObject, { a: 1, b: 2 })


     

  • 相关阅读:
    2024得物校招面试真题汇总及其解答(一)
    MySQL:事务的概念 | ACID特性 | 事务并发存在的问题 | 事务处理命令
    Xilinx约束学习笔记—— 时序约束
    STC 32位8051单片机开发实例教程 二 I/O工作模式及其配置
    SpringCloud Gateway 3.x 响应头添加 Skywalking TraceId
    Windows 10/11如何恢复永久删除的文件?
    【SQL刷题】DAY18----SQL汇总数据专项练习
    辉芒微IO单片机FT60F021-RB
    LabVIEW卡尔曼滤波技术
    【毕业设计源码】基于SSM的高校学籍信息管理系统的设计与实现
  • 原文地址:https://blog.csdn.net/m0_59359854/article/details/126240134