实现Vue3 readonly,教你如何一步步重构

本文通过实现readonly方法,一步步展示重构的流程。

前言

readonly接受一个对象,返回一个原值的只读代理。

实现 Vue3 中readonly方法,先来看一下它的使用。

<script setup>
import { readonly } from "vue";let user = {name: "wendZzoo",age: 18,address: {province: "jiangsu",city: "suzhou",},
};
const copyUser = readonly(user);user.age = 20;
copyUser.age = 18;user.address.city = "nanjing";
copyUser.address.city = "suzhou";
</script><template>{{ user }}{{ copyUser }}
</template>

readonly是原值的代理,当原值修改时候,readonly包裹的值也会被修改,但是修改readonly的值,控制台会有警告报错,且无法修改。

readonly是个代理

readonly的实现和reactive一样,只是它无法实现更新操作,意味着没有触发依赖,也就相当于不会收集依赖。

先来写一个核心逻辑的单测,新建readonly.spec.ts

import { readonly } from "../reactive";it("happy path", () => {const original = { foo: 1, bar: { bar: 2 } };const wapper = readonly(original);expect(wapper).not.toBe(original);expect(wapper.foo).toBe(1);
});

reactive.ts中导出readonly方法,

import { track, trigger } from "./effect";export function reactive(raw) {return new Proxy(raw, {get: (target, key) => {let res = Reflect.get(target, key);track(target, key);return res;},set: (target, key, value) => {let res = Reflect.set(target, key, value);trigger(target, key);return res;},});
}export function readonly(raw) {return new Proxy(raw, {get: (target, key) => {let res = Reflect.get(target, key);return res;},set: (target, key, value) => {return true;},});
}

执行单测yarn test readonly

重构

单测通过说明readonly方法的核心功能,返回一个代理,已经实现了。

但是代码中可以优化的地方有很多,让我们一步步重构,看看代码是怎么变成你不认识的样子。

抽离get函数

reactive方法和readonly方法中,将重复的地方提取出来封装成函数。

function get(target, key) {let res = Reflect.get(target, key);track(target, key);return res;
}

reactive方法和readonly方法中get操作的唯一区别就是是否调用了track,为了复用get函数,可以将其封装成一个高阶函数,传入布尔值进行判断。

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

代码一致性

那为了保证代码的一致性,get已经抽离,set也做相应的抽离封装。

function createSetter() {return function set(target, key, value) {let res = Reflect.set(target, key, value);trigger(target, key);return res;};
}

重构到这儿,reactive.ts文件变成了如下这样:

import { track, trigger } from "./effect";
function createGetter(isReadonly = false) {return function get(target, key) {let res = Reflect.get(target, key);if (!isReadonly) {track(target, key);}return res;};
}
function createSetter() {return function set(target, key, value) {let res = Reflect.set(target, key, value);trigger(target, key);return res;};
}
export function reactive(raw) {return new Proxy(raw, {get: createGetter(),set: createSetter(),});
}
export function readonly(raw) {return new Proxy(raw, {get: createGetter(true),set: (target, key, value) => {return true;},});
}

再次执行单测,验证重构是否破坏了原有功能,测试通过说明重构没有问题,继续下一步的重构。

抽离成单独文件

reactive方法和readonly方法中都存在getset,那可以优化的点就是将这块逻辑抽离。单独新建一个文件baseHandler.ts

import { track, trigger } from "./effect";
function createGetter(isReadonly = false) {return function get(target, key) {let res = Reflect.get(target, key);if (!isReadonly) {track(target, key);}return res;};
}
function createSetter() {return function set(target, key, value) {let res = Reflect.set(target, key, value);trigger(target, key);return res;};
}
export const multableHandler = {get: createGetter(),set: createSetter(),
};
export const readonlyHandler = {get: createGetter(true),set: (target, key, value) => {return true;},
};

相应的,原本reactive.ts中代码修改成:

import { multableHandler, readonlyHandler } from "./baseHandler";
export function reactive(raw) {return new Proxy(raw, multableHandler);
}
export function readonly(raw) {return new Proxy(raw, readonlyHandler);
}

这儿发现 new Proxy重复,可以将这块逻辑单独封装成一个函数,让代码的语义化更好。

import { multableHandler, readonlyHandler } from "./baseHandler";
export function reactive(raw) {return createActiveObject(raw, multableHandler);
}
export function readonly(raw) {return createActiveObject(raw, readonlyHandler);
}
function createActiveObject(raw, baseHandler) {return new Proxy(raw, baseHandler);
}

缓存

回顾代码,是否还有优化的地方?

baseHandler.ts中,每次getset都是创建一个函数,可以采用缓存,减少这样不必要的执行。

const get = createGetter();
const set = createSetter();
const readonlyGet = createGetter(true);
export const multableHandler = {get,set,
};
export const readonlyHandler = {get: readonlyGet,set: (target, key, value) => {return true;},
};

再次执行单测,验证重构是否破坏了原有功能。

返回警告报错

单测,jest.fn()模拟一个警告函数,当readonly值更新时,断言这个警告函数执行了。

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

实现上很简单,就是在readonlyset方法里打印一下告警提示。

export const readonlyHandler = {get: readonlyGet,set: (target, key, value) => {console.warn(`Set operation on key ${key} failed: target is readonly`,target);return true;},
};

最后,执行所有测试yarn test,测试通过。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/175968.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

无人机航迹规划MATLAB:七种优化算法(DBO、LO、SWO、COA、LSO、KOA、GRO)求解无人机路径规划

一、七种算法&#xff08;DBO、LO、SWO、COA、LSO、KOA、GRO&#xff09;简介 1、蜣螂优化算法DBO 蜣螂优化算法&#xff08;Dung beetle optimizer&#xff0c;DBO&#xff09;由Jiankai Xue和Bo Shen于2022年提出&#xff0c;该算法主要受蜣螂的滚球、跳舞、觅食、偷窃和繁…

php 插入排序算法实现

插入排序是一种简单直观的排序算法&#xff0c;它的基本思想是将一个数据序列分为有序区和无序区&#xff0c;每次从无序区选择一个元素插入到有序区的合适位置&#xff0c;直到整个序列有序为止 5, 3, 8, 2, 0, 1 HP中可以使用以下代码实现插入排序算法&#xff1a; functi…

Scala---方法与函数

一、Scala方法的定义 有参方法&无参方法 def fun (a: Int , b: Int) : Unit {println(ab) } fun(1,1)def fun1 (a: Int , b: Int) ab println(fun1(1,2)) 注意点&#xff1a; 方法定义语法 用def来定义可以定义传入的参数&#xff0c;要指定传入参数的类型方法可以写返…

西门子精彩触摸屏SMART LINE V4 面板使用U盘下载项目程序的具体方法示例

西门子精彩触摸屏SMART LINE V4 面板使用U盘下载项目程序的具体方法示例 WinCC flexible SMART V4 SP1 软件针对SMART LINE V4 面板新增了使用U盘下载项目功能。 注意:“使用U盘下载项目”功能仅支持触摸屏OS版本为V4.0.1.0 及以上的设备。 使用U盘下载项目的步骤可参考以下内…

Skybox天空盒子的更换教程_unity基础开发教程

Skybox天空盒子的更换 Skybox的下载与导入更换SkyboxSkybox属性自定义 Skybox的下载与导入 打开资源商店 搜索FREE Skybox 这里是我使用的是这一款资源&#xff0c;点击添加至我的资源 打开包管理器Package Manager Packages选择My Assets 搜索Sky 选择刚刚添加的天空盒子 点…

高质量实时渲染笔记

文章目录 Real-time shadows1 自遮挡问题2 解决阴影detach问题&#xff1f;3 Aliasing4 近似积分5 percentage closer soft shadows(PCSS)percenta closer filtering(PCF)PCSS的思想 6 Variance Soft Shadow Mapping (VSSM)步骤Moment Shadow Mapping 7 Distance field shadow …

Leetcode刷题详解—— 图像渲染

1. 题目链接&#xff1a;733. 图像渲染 2. 题目描述&#xff1a; 有一幅以 m x n 的二维整数数组表示的图画 image &#xff0c;其中 image[i][j] 表示该图画的像素值大小。 你也被给予三个整数 sr , sc 和 newColor 。你应该从像素 image[sr][sc] 开始对图像进行 上色填充 。…

PHP使用文件缓存实现html静态化

<?php // 动态生成的内容 $content "<html><body><h1>time:".date("Y-m-d H:i:s")."</h1></body></html>"; // 静态文件保存路径和文件名 $staticFilePath "file.html"; if(file_exists($s…

3D Gaussian Splatting文件的压缩【3D高斯泼溅】

在上一篇文章中&#xff0c;我开始研究高斯泼溅&#xff08;3DGS&#xff1a;3D Gaussian Splatting&#xff09;。 它的问题之一是数据集并不小。 渲染图看起来不错。 但“自行车”、“卡车”、“花园”数据集分别是一个 1.42GB、0.59GB、1.35GB 的 PLY 文件。 它们几乎按原样…

【算法】算法题-20231110

一、力口&#xff1a;506. 相对名次 简单 给你一个长度为 n 的整数数组 score &#xff0c;其中 score[i] 是第 i 位运动员在比赛中的得分。所有得分都 互不相同 。 运动员将根据得分 决定名次 &#xff0c;其中名次第 1 的运动员得分最高&#xff0c;名次第 2 的运动员得分第…

图论16-拓扑排序

文章目录 1 拓扑排序2 拓扑排序的普通实现2.1 算法实现 - 度数为0入队列2.2 拓扑排序中的环检测 3 深度优先遍历的后续遍历3.1 使用环检测类先判断是否有环3.2 调用无向图的深度优先后续遍历方法&#xff0c;进行DFS 1 拓扑排序 对一个有向无环图G进行拓扑排序&#xff0c;是将…

部署百川大语言模型Baichuan2

Baichuan2是百川智能推出的新一代开源大语言模型&#xff0c;采用 2.6 万亿 Tokens 的高质量语料训练。在多个权威的中文、英文和多语言的通用、领域 benchmark 上取得同尺寸最佳的效果。包含有 7B、13B 的 Base 和 Chat 版本&#xff0c;并提供了 Chat 版本的 4bits 量化。 模…