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)

})
相关推荐
玫城5 小时前
[ VUE ] 封装通用数组校验组件,el-input内使用
前端·javascript·vue.js
弓.长.9 小时前
React Native 鸿蒙跨平台开发:实现一个多功能单位转换器
javascript·react native·react.js
南半球与北海道#9 小时前
前端打印(三联纸票据打印)
前端·vue.js·打印
摘星编程9 小时前
React Native for OpenHarmony 实战:ToggleSwitch 切换开关详解
javascript·react native·react.js
董世昌419 小时前
深入浅出 JavaScript 常用事件:从原理到实战的全维度解析
前端
满栀5859 小时前
分页插件制作
开发语言·前端·javascript·jquery
qq_4061761410 小时前
深入剖析JavaScript原型与原型链:从底层机制到实战应用
开发语言·前端·javascript·原型模式
弓.长.10 小时前
React Native 鸿蒙跨平台开发:BottomSheet 底部面板详解
javascript·react native·react.js
开开心心_Every10 小时前
免费窗口置顶小工具:支持多窗口置顶操作
服务器·前端·学习·macos·edge·powerpoint·phpstorm
摘星编程10 小时前
React Native for OpenHarmony 实战:Permissions 权限管理详解
javascript·react native·react.js