vue2 响应式原理

变化侦测

Vue.js最独特的特性之一是看起来并不显眼的响应式系统。当数据变化时 ,UI界面跟着变化,所以平时开发只需要关心数据 ,包括样式也是数据。简单来说变化侦测的作用就是着侦测数据的变化,当数据变化时,会通知视图相应的更新。

如何追踪变化

可以通过 Object.defineProperty来监测对象变化 ,Object.defineProperty用法如下

定义:: 静态方法Object.defineProperty()直接在对象上定义一个新属性,或者修改对象上已有的属性,并返回该对象。
Object.defineProperty(obj, prop, descriptor) obj 定义的对象 prop一个字符串 descriptor 被定义或修改的属性的描述符。

kotlin 复制代码
Object.defineProperty(this, 'message', {
        get: function() {
                      console.log('Getting value:', this._message);
                        return this._message;
                    },
       set: function(newValue) {
                        console.log('Setting value to:', newValue);
                        if (newValue !== this._message) {
                            this._message = newValue;
                            this.$forceUpdate(); // 强制更新视图
                        }
                    }
                });

如何收集依赖 Dep

该模板中使用了数据name,所以当它发生变化时,要向使用了它的地方发送通知。 对于上面的问题是,先收集依赖,用到name的地方收集起来 ,然后等属性发生变化时,把之前收集好的依赖循环触发一遍 总结: 在get中收集依赖 在set中触发依赖

依赖收集在哪里

封装一个dep 类

javascript 复制代码
class Dep {
  constructor() {
    // 存储订阅者的集合
    this.subscribers = new Set();
  }

  // 将当前活动的副作用添加到订阅者集合中
  depend() {
    if (activeEffect) {
      this.subscribers.add(activeEffect);
    }
  }

  // 通知所有订阅者重新运行他们的副作用
  notify() {
    this.subscribers.forEach(effect => effect());
  }
}

依赖是谁 Watch

观察者,当依赖的数据变化时,触发回调函数。

kotlin 复制代码
// Watcher 类:观察者,当依赖的数据变化时,触发回调函数
class Watcher {
  constructor(vm, expOrFn, cb) {
    this.vm = vm;
    this.getter = parsePath(expOrFn);
    this.cb = cb;
    this.value = this.get();
  }

  // 获取值,并进行依赖收集
  get() {
    Dep.target = this;
    const value = this.getter.call(this.vm, this.vm);
    Dep.target = null;
    return value;
  }

  // 更新值
  update() {
    const oldValue = this.value;
    this.value = this.get();
    this.cb.call(this.vm, this.value, oldValue);
  }
}

watcher 是如何与模版关联的

Watcher 主要用于监听数据的变化,并在数据变化时更新视图。为了将 Watcher 与模板关联起来,Vue 使用了一个编译器将模板转换为渲染函数(render function),然后通过这些渲染函数来创建虚拟 DOM(VNode)。当数据发生变化时,Watcher 会触发重新渲染过程,更新实际的 DOM。 一个dom 关联一个watcher 创建相应的 Watcher 实例,并将初始值插入到 DOM 中

Array的变化侦测

获取原始数组原型:

ini 复制代码
const arrayProto = Array.prototype;

创建新的数组方法对象:

ini 复制代码
const arrayMethods = Object.create(arrayProto);

定义需要拦截的方法:

ini 复制代码
const methodsToPatch = [
  'push',
  'pop',
  'shift',
  'unshift',
  'splice',
  'sort',
  'reverse'
];

拦截每个方法并添加响应性逻辑

复制代码
ini 复制代码
methodsToPatch.forEach(function (method) {
  const original = arrayProto[method];
  def(arrayMethods, method, function mutator(...args) {
    const result = original.apply(this, args);
    const ob = this.__ob__;
    let inserted;
    switch (method) {
      case 'push':
      case 'unshift':
        inserted = args;
        break;
      case 'splice':
        inserted = args.slice(2);
        break;
    }
    if (inserted) ob.observeArray(inserted);
    ob.dep.notify();
    return result;
  });
});

它的作用是将拦截器(加工后具备拦截功能的arrayMethods)赋值给value.proto ,通过 proto 可以很巧妙地实现覆盖value原型的功能

kotlin 复制代码
 export class Observer {
         constructor (value) {
         this.value = value
         }
 
 if (Array.isArray(value)) {
  value.__proto__ = arrayMethods
  } else {
 this.walk(value) 
      } 
      } 
      }
相关推荐
一个努力的小码农7 小时前
Rust中if let与while let语法糖的工程哲学
前端·rust
雾岛听风来7 小时前
Android开发中常用高效数据结构
前端·javascript·后端
IT_陈寒7 小时前
Vue 3性能优化实战:这5个Composition API技巧让你的应用快30%
前端·人工智能·后端
IT_陈寒7 小时前
Vue3性能翻倍的5个秘密:从Composition API到Tree Shaking实战指南
前端·人工智能·后端
IT_陈寒7 小时前
JavaScript 性能优化:3个V8引擎隐藏技巧让你的代码提速50%
前端·人工智能·后端
沐怡旸7 小时前
【技术选型】前端框架:Vue vs React - 组合式API与Hooks的哲学之争
前端·面试
charlie1145141917 小时前
HTML 理论系统笔记2
前端·笔记·学习·html·基础·1024程序员节·原生
2501_938780287 小时前
Ionic + Angular 跨端实战:用 Capacitor 实现相机拍照功能并适配移动端
前端·数码相机·angular.js
m0_64880493_江哥8 小时前
Python实现随机选播视频的示例代码
前端·python·音视频