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)

})
相关推荐
范文杰3 分钟前
AI 时代如何更高效开发前端组件?21st.dev 给了一种答案
前端·ai编程
拉不动的猪11 分钟前
刷刷题50(常见的js数据通信与渲染问题)
前端·javascript·面试
拉不动的猪19 分钟前
JS多线程Webworks中的几种实战场景演示
前端·javascript·面试
FreeCultureBoy1 小时前
macOS 命令行 原生挂载 webdav 方法
前端
uhakadotcom2 小时前
Astro 框架:快速构建内容驱动型网站的利器
前端·javascript·面试
uhakadotcom2 小时前
了解Nest.js和Next.js:如何选择合适的框架
前端·javascript·面试
uhakadotcom2 小时前
React与Next.js:基础知识及应用场景
前端·面试·github
uhakadotcom2 小时前
Remix 框架:性能与易用性的完美结合
前端·javascript·面试
uhakadotcom2 小时前
Node.js 包管理器:npm vs pnpm
前端·javascript·面试
LaoZhangAI3 小时前
2025最全GPT-4o图像生成API指南:官方接口配置+15个实用提示词【保姆级教程】
前端