ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

Vue3.0响应式实现

2019-12-02 17:51:46  阅读:228  来源: 互联网

标签:function return target 实现 effect 响应 Vue3.0 let key


基于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';

  

标签:function,return,target,实现,effect,响应,Vue3.0,let,key
来源: https://www.cnblogs.com/QQPrincekin/p/11972142.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有