zoukankan      html  css  js  c++  java
  • Vue3.0响应式实现

    基于Proxy

    // 弱引用映射表 es6 防止对象不能被回收
    let toProxy = new WeakMap(); // 原对象: 代理过得对象
    let toRaw = new WeakMap(); // 被代理过的对象: 原对象
    
    // 判断为对象
    function isObject(val) {
        return typeof val === 'object' && val !== null
    }
    // 区分改变数组长度还是数值
    function hasOwn(target, key) {
        return target.hasOwnProperty(key)
    }
    
    function reactive(target) {
        // 创建响应式对象
        return createReactiveObject(target)
    }
    // 创建响应式对象
    function createReactiveObject(target) {
        if (!isObject(target)) {
            return target
        }
        let proxy = toProxy.get(target); // 如果已经代理过 直接返回代理结果
        if (proxy) {
            return proxy;
        }
        if (toRaw.has(target)) { // 防止对象被多次代理
            return target;
        }
        let baseHandler = {
            // Reflect优点: 不报错 有返回值 会替代Object 上的方法
            // Proxy + reflect 反射
            get(target, key, receiver) {
                // console.log('查询');
                let result = Reflect.get(target, key, receiver);
                // 收集依赖订阅 把当前key 和effect对应
                track(target, key);
                // result 当前获取到的值
                return isObject(result) ? reactive(result) : result; // 深层次代理 多层代理 (递归)
            },
            set(target, key, value, receiver) {
                // console.log('设置');
                // 区分改变数组长度还是数值
                let hadKey = hasOwn(target, key);
                let oldValue = target[key];
                let res = Reflect.set(target, key, value, receiver);
                if (!hadKey) {
                    trigger(target, 'add', key);
                    // console.log('新增属性')
                } else if (oldValue !== value) { // 表示属性修改
                    // console.log('修改属性')
                    trigger(target, 'set', key);
                }//
                // 设置成功返回值
                return res;
            },
            deleteProperty(target, key) {
                // console.log('删除');
                let res = Reflect.deleteProperty(target, key)
                return res;
            }
        }
        let observed = new Proxy(target, baseHandler);
        toProxy.set(target, observed);
        toRaw.set(observed, target);
        return observed
    }
    
    // let proxy = reactive({name: {n: 'wyq'}});
    // proxy.name.n = '王瘦瘦'
    // console.log(proxy.name.n)
    // let arr = [1, 2, 3];
    // let proxy = reactive(arr)
    // proxy.length = 100;
    
    // 发布订阅模式
    // 栈结构
    let activeEffectStacks = []; // 栈型结果
    let targetsMap = new WeakMap(); // 集合和哈希表
    function track(target, key) { // target中的key变化 执行数组方法
        let effect = activeEffectStacks[activeEffectStacks.length - 1];
        if (effect) { // 有对应关系 创建关联
            let depsMap = targetsMap.get(target);
            if (!depsMap) {
                targetsMap.set(target, depsMap = new Map());
            }
            let deps = depsMap.get(key);
            if (!deps) {
                depsMap.set(key, deps = new Set());
            }
            if (!deps.has(effect)) {
                deps.add(effect);
            }
            // 动态创建依赖关系
        }
        // 不管
    }
    
    function trigger(target, type, key) {
        let depsMap = targetsMap.get(target);
        if (depsMap) {
            let deps = depsMap.get(key);
            if (deps) { // 当前key 对应effect 依次执行
                deps.forEach((effect) => {
                    effect();
                })
            }
        }
    }
    
    // 响应式 副作用
    function effect(fn) {
        // 把fn包装成响应式函数
        let effect = createReactiveEffect(fn);
        effect(); // 先执行一次
    }
    
    function createReactiveEffect(fn) {
        // 高阶函数
        let effect = function () { // is 创建的响应式的effect
            return run(effect , fn); // 1.让fn执行 2.存入栈
        };
        return effect;
    }
    
    function run(effect, fn) {
        try {
            activeEffectStacks.push(effect);
            fn();
        }finally {
            activeEffectStacks.pop();
        }
    }
    
    let obj = reactive({name: 'wyq'});
    effect(() => { // effect默认执行两次 默认先执行一次 依赖数据变化在执行一次
       console.log(obj.name)
    });
    obj.name = '王瘦瘦';
    obj.name = '王瘦瘦';
    obj.name = 'SpongeBob';
    

      

  • 相关阅读:
    java得到当前日期的前一天或后一天
    java通过年月得到该月每一天的日期
    前后端分离的跨域请求问题解决
    关于java中分割字符串
    Linux终端命令
    eclipse中tomcat可以start启动,无法debug启动的解决
    通过DOS界面查看电脑上端口使用情况
    前端css
    mysql数据库
    IO阻塞
  • 原文地址:https://www.cnblogs.com/QQPrincekin/p/11972142.html
Copyright © 2011-2022 走看看