js高阶-响应式原理

what

  • 当变量变化的时候,涉及到这个变量的操作重新执行
  • 更多是针对对象类型进行监听(原始数据类型是不变的,任何的修改实质是创建了新值)

how

  • 响应式对象创建 Proxy
  • 响应式对象与依赖之间的映射关系 weakMap-map-set
  • 依赖函数收集

创建响应式对象

vue3

js 复制代码
//响应式对象创建

function reactive(obj) {

  return new Proxy(obj, {

    get(target, key) {

      const depend = getDepend(target, key);

      //依赖收集

      depend.addDepend();

      //返回

      return Reflect.get(target, key);

    },

    set(target, key, value) {

      const depend = getDepend(target, key);

      depend.notify();

      Reflect.set(target, key, value);

  

    },

  });

}

vue2方式

js 复制代码
//vue2的响应式操作

function reactive2(obj){

    Object.keys(obj).forEach(key=>{

        let value = obj[key]

        Object.defineProperty(obj,key,{

            get(){

                const depend = getDepend(obj,key)

                depend.depend()

                return value

            },

            set(newValue){

                const depend = getDepend(obj,key)

                depend.notify()

                value = newValue

            }

        })

    })

  

    return obj

}

对象-依赖关系

js 复制代码
let targetMap = new WeakMap(); // 响应式对象与依赖关系的映射 1.对象与属性的映射 2.属性与依赖函数的映射

  

class Depend {

  constructor() {

    this.reactiveFns = new Set();

  }

  //收集依赖

  addDepend() {

    if (activeReactiveFn) {

      this.reactiveFns.add(activeReactiveFn);

    }

  }

  //通知依赖

  notify() {

    this.reactiveFns.forEach((fn) => {

        if(fn){

          fn()

        }

    });

  }

}

function getDepend(target, key) {

  //1.根据target获取map

  let map = targetMap.get(target);

  if (!map) {

    map = new Map();

    targetMap.set(target, map);

  }

  //2.根据key获取depend

  let depend = map.get(key);

  if (!depend) {

    depend = new Depend();

    map.set(key, depend);

  }

  return depend;

}

依赖函数收集

js 复制代码
let activeReactiveFn = null; // 当前激活的响应式函数
//响应式函数收集---被watchFn收集的函数会成为响应式的依赖,当数据变化这些函数会被调用

function watchFn(fn) {

  activeReactiveFn = fn;

  fn();

  activeReactiveFn = null;

}

使用

js 复制代码
const obj = {

  name: "张三",

  age: 18,

};

//1.创建响应式对象

const objRef = reactive(obj);

//2.收集依赖函数

watchFn(()=>{

    console.log(objRef.name)

})

watchFn(()=>{

    console.log(objRef.age)

})
相关推荐
逆旅行天涯24 分钟前
【vitePress】基于github快速添加评论功能(giscus)
前端·github
我有一棵树42 分钟前
style标签没有写lang=“scss“引发的 bug 和反思
前端·bug·scss
陈奕迅本讯2 小时前
HTML5和CSS3拔高
前端·css3·html5
USER_A0012 小时前
JavaScript笔记进阶篇01——作用域、箭头函数、解构赋值
javascript·笔记
画船听雨眠aa2 小时前
vue项目创建与运行(idea)
前端·javascript·vue.js
我想学LINUX2 小时前
【2024年华为OD机试】(C/D卷,200分)- 5G网络建设 (JavaScript&Java & Python&C/C++)
java·c语言·javascript·网络·python·5g·华为od
大码猴2 小时前
用好git的几个命令,领导都夸你干的好~
前端·后端·面试
℡52Hz★2 小时前
如何正确定位前后端bug?
前端·vue.js·vue·bug
小小弯_Shelby2 小时前
vue+arcgis api for js实现地图测距的分段统计线段长度
vue.js·arcgis