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

  

最新文章

  1. [Spring]04_最小化Spring XML配置
  2. 基于Dubbo框架构建分布式服务(二)
  3. 使用BOOTICE 恢复系统启动项
  4. oop、try_except、单例模式
  5. SQL Server 2012 创建数据库快照
  6. 2014年2月份第4周51Aspx源码发布详情
  7. 关于 mysql 2003 客户端连接报错的处理方法
  8. josn 转php
  9. [Leetcode][Python]49: Anagrams
  10. php 汉字转换成拼音
  11. 获取ini文件所有的Sections和Keys
  12. iOS 之使用CAShapeLayer中的CAGradientLayer实现圆环的颜色渐变
  13. hdu 4919 Exclusive or
  14. Windows程序设计学习笔记(五)——菜单资源和加速键的使用
  15. Ruby数组(1):基本用法
  16. EF CodeFirst系列(2)---CodeFirst的数据库初始化
  17. Vim怎么批量处理文件将tab变为space
  18. xinetd服务
  19. centOS redis的安装及配置
  20. Docker(三):Dockerfile 命令详解

热门文章

  1. C# asp.net XML格式的字符串显示不全
  2. vue实现PC端调用摄像头拍照人脸录入、移动端调用手机前置摄像头人脸录入、及图片旋转矫正、压缩上传base64格式/文件格式
  3. MySQL系列理论知识
  4. Taro -- 微信小程序wxParse达到html转换wxml
  5. [BJWC2010]严格次小生成树(LCA,最小生成树)
  6. 前端经典布局:Sticky footer 布局
  7. LeetCode--050--Pow(x,n)
  8. 【GMOJ6377】幽曲[埋骨于弘川]
  9. 使用 pyenv 管理 Python 版本
  10. 关于富文本复制word,里面掺杂图片上传的问题