Vue3源码reactive和readonly对象嵌套转换,及实现shallowReadonly

news/2024/7/10 1:11:20 标签: 前端, vue, javascript

前言

官方文档中对reactive的描述:

响应式转换是“深层”的:它会影响到所有嵌套的属性。一个响应式对象也将深层地解包任何 ref 属性,同时保持响应性。

官方文档中对readonly的描述:

只读代理是深层的:对任何嵌套属性的访问都将是只读的。它的 ref 解包行为与 reactive() 相同,但解包得到的值是只读的。

这意味着嵌套对象内的对象拥有和原对象一样的功能。

简单的来实践测试一下:

vue"><script setup>
import { isReactive, isReadonly, reactive, readonly } from "vue";

const original = reactive({
  nested: {
    foo: 1,
  },
  array: [{ bar: 2 }],
});

const copy = readonly(original);
</script>

<template>
  {{ isReactive(original.nested) }}
  {{ isReactive(original.array) }}
  {{ isReactive(original.array[0]) }}

  {{ isReadonly(copy.nested) }}
</template>

页面中显示四个true

那测试用例可以完善一下,测试之前实现的代码是否支持这样的功能。

reactive

reactive.spec.ts中添加测试用例:

it("nested reactive", () => {
  const original = reactive({
    nested: { foo: 1 },
    array: [{ bar: 2 }],
  });
  expect(isReacive(original.nested)).toBe(true);
  expect(isReacive(original.array)).toBe(true);
  expect(isReacive(original.array[0])).toBe(true);
});

执行单测,如期不通过。

实现

实现思路,访问嵌套对象内对象时候,将内部对象也实现响应式,即调用reactive方法。那前提是访问的是对象,则需要添加一个是否是对象的判断。

import { isObject } from "../shared";

function createGetter(isReadonly = false) {
  return function get(target, key) {
    let res = Reflect.get(target, key);

    if (key === ReactiveFlags.IS_REACTIVE) {
      return !isReadonly;
    } else if (key === ReactiveFlags.IS_READONLY) {
      return isReadonly;
    }

    if (isObject(res)) {
      return reactive(res);
    }

    if (!isReadonly) {
      track(target, key);
    }
    return res;
  };
}

在公共方法中导出isObject方法,

export function isObject(val) {
  return val !== null && typeof val === "object";
}

执行单测yarn test reactive

readonly

readonly.spec.ts中完善对嵌套对象的断言,

it("happy path", () => {
  const original = { foo: 1, bar: { baz: 2 } };
  const wapper = readonly(original);

  expect(wapper).not.toBe(original);
  expect(wapper.foo).toBe(1);

  expect(isReadonly(wapper)).toBe(true);
  expect(isReadonly(original)).toBe(false);
  // 判断嵌套对象
  expect(isReadonly(wapper.bar)).toBe(true);
});

执行单测yarn test readonly,预期不通过。

实现

readonly的实现和reactive一致。

if (isObject(res)) {
  return isReadonly ? readonly(res) : reactive(res);
}

执行单测,

最后再执行全部测试用例,验证是否对其他功能存在影响。

shallowReadonly

如果你在上文中点进了官方文档的链接,在readonly的详细描述中:

要避免深层级的转换行为,请使用 shallowReadonly() 作替代。

readonly()不同,shallowReadonly没有深层级的转换:只有根层级的属性变为了只读。属性的值都会被原样存储和暴露,这也意味着值为ref的属性不会被自动解包了。

单测

新增shallowReadonly.spec.ts

shallowReadonly是浅层转换,还是用isReadonly进行检测,最外层的是可读的,深层次的不是。

it("should not make non-reactive properties reactive", () => {
  const original = reactive({ foo: 1, bar: { baz: 2 } });

  const wapper = shallowReadonly(original);
  expect(isReadonly(wapper)).toBe(true);
  expect(isReadonly(wapper.bar)).toBe(false);
});

实现

reactive.ts导出shallowReadonly方法,和readonly类似,只是handler参数不同,

export function shallowReadonly(raw) {
  return createActiveObject(raw, shallowReadonlyHandler);
}

baseHandler.ts导出shallowReadonlyHandler对象,

export const shallowReadonlyHandler = {}

handlerget操作都是根据createGetter方法生成,那shallowReadnoly的特点是最外层的是可读的,深层次的不是,可以像readonly一样添加一个参数来判断,如果是shallow直接返回结果,也不需要进行深层次的转换和track

function createGetter(isReadonly = false, shallow = false) {
  return function get(target, key) {
    let res = Reflect.get(target, key);

    if (key === ReactiveFlags.IS_REACTIVE) {
      return !isReadonly;
    } else if (key === ReactiveFlags.IS_READONLY) {
      return isReadonly;
    }

    if (shallow) {
      return res;
    }

    if (isObject(res)) {
      return isReadonly ? readonly(res) : reactive(res);
    }

    if (!isReadonly) {
      track(target, key);
    }
    return res;
  };
}

全局定义:

const shallowReadonlyGet = createGetter(true, true);

shallowReadonlyHandler对象和readonlyHandler相似,只是get不同,可以复用之前实现的extend方法。

export const shallowReadonlyHandler = extend({}, readonlyHandler, {
  get: shallowReadonlyGet,
});

那可以顺便验证shallowReadonly的更新操作的测试用例,

it("should call console warn when call set", () => {
  console.warn = jest.fn();
  const original = shallowReadonly({ foo: 1 });
  original.foo = 2;

  expect(console.warn).toHaveBeenCalled();
});

执行单测yarn test shallowReadonly

总结

  1. reactive响应式转换是“深层”的:它会影响到所有嵌套的属性。readonly只读代理是深层的,对任何嵌套属性的访问都将是只读的。
  2. 获取对象深层次嵌套,会触发get操作,在判断该嵌套目标也是对象类型时就可以再次用reactivereadonly包裹返回。
  3. shallowReadonly就是在深层次转换和track逻辑之前返回结果即可。

http://www.niftyadmin.cn/n/5183686.html

相关文章

2023年05月 Python(五级)真题解析#中国电子学会#全国青少年软件编程等级考试

Python等级考试(1~6级)全部真题・点这里 一、单选题(共25题,每题2分,共50分) 第1题 有列表L=[‘UK’,‘china’,‘lili’,“张三”],print(L[-2])的结果是?( ) A: UK B: ‘lili’,‘张三’ C: lili D: ‘UK’,‘china’,‘lili’ 答案:C 列表元素定位 第2题 …

一、PMP初识

PMP初识 一、什么是项目 定义&#xff1a;项目是为创造独特的产品、服务或成果而进行的临时性工作 项目的特点 1&#xff09;独特性&#xff1a; 实现项目目标可能会产生以下一个或者多个独特的产品、成功、服务或者组合&#xff1b; 2&#xff09;渐进明细&#xff1a;随…

JVM虚拟机——类加载器(JDK8及以前,打破双亲委派机制)(JDK9之后的类加载器)

目录 1.自定义类加载器2.线程上下文类加载器3.OSGi模块化4.JDK9之后的类加载器5.类加载器总结 1.自定义类加载器 ⚫ 一个Tomcat程序中是可以运行多个Web应用的&#xff0c;如果这两个应用中出现了相同限定名的类&#xff0c;比如Servlet类&#xff0c;Tomcat要保证这两个类都能…

centos7安装cuda和nvidia-driver

文章目录 前言1.检查系统有没有cuda支持的GPU2.安装gcc3.下载cuda4.内核头文件和开发包安装5.安装EPEL依赖6.安装cuda7.重启8.查看显卡信息前言 如果不是 centos7 的 linux,可以直接参考官方文档安装其他系统的 cuda,但是文档很长需要耐心 ↓ https://docs.nvidia.com/cuda/…

lxml基本使用

lxml是python的一个解析库&#xff0c;支持HTML和XML的解析&#xff0c;支持XPath解析方式&#xff0c;而且解析效率非常高 XPath&#xff0c;全称XML Path Language&#xff0c;即XML路径语言&#xff0c;它是一门在XML文档中查找信息的语言&#xff0c;它最初是用来搜寻XML文…

【设计原则篇】聊聊接口隔离原则

是什么 客户端不应该强迫依赖它不需要的接口&#xff0c;客户端可以理解为接口的调用者或者使用者。接口其实就是一种规范&#xff0c;比如手机都是同样的接口&#xff0c;那么充电器就可以使用。 接口从细的层面来说的话&#xff0c;其实分三种&#xff0c;一组API接口集合、单…

文件包含学习笔记总结

文件包含概述 ​ 程序开发人员通常会把可重复使用函数或语句写到单个文件中&#xff0c;形成“封装”。在使用某个功能的时候&#xff0c;直接调用此文件&#xff0c;无需再次编写&#xff0c;提高代码重用性&#xff0c;减少代码量。这种调用文件的过程通常称为包含。 ​ 程…

基于单片机的水位检测系统仿真设计

**单片机设计介绍&#xff0c; 基于单片机的水位检测系统仿真设计 文章目录 一 概要二、功能设计设计思路 三、 软件设计原理图 五、 程序六、 文章目录 一 概要 基于单片机的水位检测系统仿真系统是一种用于模拟水位检测系统的工作过程&#xff0c;以验证设计方案的可行性和优…