VUE设计与实现共读系列之ref的实现【响应式原理】

前言

我们先顺一下vue使用响应式数据的流程:

vue 是通过 refreactive 来创建响应式值,改变响应式值,视图跟着发生变化。

我们今天就来看一下refreactive是如何实现的

准备

首先,打开ref函数的位置

我们可以看到一个被ref包裹的响应式数据,其实是一个RefImpl对象

1. 创建ref

js 复制代码
export function ref(value) {
  return createRef(value);
}

function createRef(value) {
  const refImpl = new RefImpl(value);

  return refImpl;
}

可以看到创建一个ref对象的本质就是创建一个RefImpl装饰对象

2. 编写RefImpl对象

js 复制代码
export class RefImpl {
  private _rawValue: any; // 原值
  private _value: any;	// 代理值
  public dep;	// 依赖数组:用来存放依赖
  public __v_isRef = true; // 区分是否是ref这个对象类型

  constructor(value) {
    this._rawValue = value;
    // 看看value 是不是一个对象,如果是一个对象的话
    // 那么需要用 reactive 包裹一下
    this._value = convert(value);
    this.dep = createDep();  // 创建effect对象
  }

  get value() {
    // 收集依赖
    trackRefValue(this);
    return this._value;
  }

  set value(newValue) {
    // 当新的值不等于老的值的话,
    // 那么才需要触发依赖
    if (hasChanged(newValue, this._rawValue)) {
      // 更新值
      this._value = convert(newValue);
      this._rawValue = newValue;
      // 触发依赖
      triggerRefValue(this);
    }
  }
}

❤️❤️ 为什么要有_rawValue_value呢?

答:_value用来保存我们加工后的具有响应式的对象,_rawValue保存的是原始的值

3. 依赖收集和触发

依赖收集

js 复制代码
// trackRefValue
export function trackRefValue(ref) {
  if (isTracking()) { // 判定师傅可以进行收集
    trackEffects(ref.dep);
  }
}

// trackEffects
let activeEffect = void 0;

export function trackEffects(dep) {
  // 用 dep 来存放所有的 effect
  if (!dep.has(activeEffect)) {
    dep.add(activeEffect);
    (activeEffect as any).deps.push(dep);
  }
}
  • activeEffect:作用是用一个全局变量存储被注册的副作用函数

依赖触发

js 复制代码
// ref.ts
export function triggerRefValue(ref) {
  triggerEffects(ref.dep);
}

// effect.ts
export function triggerEffects(dep) {
  // 执行收集到的所有的 effect 的 run 方法
  for (const effect of dep) {
    if (effect.scheduler) {
      // scheduler 可以让用户自己选择调用的时机
      // 这样就可以灵活的控制调用了
      // 在 runtime-core 中,就是使用了 scheduler 实现了在 next ticker 中调用的逻辑
      effect.scheduler();
    } else {
      effect.run();
    }
  }
}

思考

  1. ref创建的值 .value有什么用?
相关推荐
疯狂的魔鬼1 分钟前
一套 Schema 驱动四视图:记 useCrudSchemas 的设计与实践
前端·javascript·typescript
风骏时光牛马4 分钟前
大模型开发工具高频故障与实操问题汇总代码案例大全
前端
没落英雄9 分钟前
2. 让 Agent 能读写文件、执行命令 —— LocalShellBackend 实战
前端·人工智能·架构
白雾茫茫丶9 分钟前
探索 Nuxt.js 全栈能力:用 Better-Auth 打造类型安全的 RBAC 权限系统
前端·vue.js·nuxt.js
奇奇怪怪的18 分钟前
检索增强——混合检索、Re-rank 与 Query 优化
前端
user622298649258122 分钟前
React 常用技术知识全景:从组件到 Hooks 的系统理解
前端
麻辣凉茶22 分钟前
给阿嬤一封来自云端的信(上)
前端·node.js
前端缘梦23 分钟前
LangGraph 实战:从 0 到 1 构建 AI 代码生成工作流
前端·程序员·全栈
weedsfly24 分钟前
栈和堆:JavaScript 内存的“旅馆”和“仓库”
前端·javascript·面试