Vue 3 中的 readonly 和 shallowReadonly:保护数据不被修改

Vue 3 中的 readonly 和 shallowReadonly:保护数据不被修改

在开发中,我们经常需要共享数据,但是我们并不希望共享的数据被意外修改;Vue3提供了readonly和shallowReadonly这两个API,用于创建只读的响应式对象。它们可以包装ref,reactive或者其他的响应式对象,返回一个只读的代理;

1.readonly深度只读

readonly会被传入的响应式数据转换为深度只读的代理。这个代理的任何修改都会被警告

vue 复制代码
<template>
  <h2>{{ copy }}</h2>
  <button @click="ChangeCopy">修改年龄</button>
</template>
<script setup>
import { readonly, reactive } from 'vue';
const original = reactive({
  user: {
    name: 'Alice',
    age: 20
  }
})

const copy = readonly(original)

function ChangeCopy() {
  copy.user.age = 22;
}

</script>
<style></style>

readonly在实际中有什么作用呢?

比如父组件传递配置给子组件,但是禁止子组件修改

vue 复制代码
//父组件
<template>
  <div>
    <h1>系统配置</h1>
    <p>主题:{{ config.theme }}</p>
    <p>语言:{{ config.language }}</p>
    <Child :config="readonlyConfig" />
  </div>
</template>
<script setup>
import { readonly, reactive } from 'vue';
import Child from './Child.vue';
//可修改的系统配置
const config = reactive({
  theme: 'dark',
  language: 'zh-CN',
  features: {
    bata: true,
    experimental: false
  }
})

//创建只读版本传递给子组件
const readonlyConfig = readonly(config);

//父组件可以正常修改配置
function updateConfig() {
  config.theme = 'light',
    config.language = 'en-US'
}


</script>
<style></style>
vue 复制代码
//子组件
<template>
    <div>
        <h3>子组件-只读配置</h3>
        <p>主题:{{ config.theme }}</p>
        <p>语言:{{ config.language }}</p>
        <button @click="tryModify">尝试修改配置</button>
    </div>
</template>
<script setup>
import { defineProps } from 'vue';
const props = defineProps(['config'])
function tryModify() {
    props.config.theme = 'light'
}
</script>

2.shallowReadonly渐层只读

shallowReadonly只将对象的顶层属性设为只读,嵌套的对象仍然是可变的

vue 复制代码
<template>
  <h2>{{ original }}</h2>
  <button @click="changeCount">修改顶层属性count</button>
  <button @click="changeName">修改嵌套属性name</button>
</template>
<script setup>
import { shallowReadonly, reactive } from 'vue'

const original = reactive({
  count: 0,
  user: { name: 'Alice' }
})

const copy = shallowReadonly(original)

// 修改顶层属性会失败
function changeCount() {
  copy.count = 1  // ❌ 警告
}

function changeName() {
  // 修改嵌套属性允许
  copy.user.name = 'Bob'  // ✅ 有效,且会触发响应式更新
}


</script>
<style></style>

其实shallowReadonly在现实中用的并不多,这里还展示一种实际使用案例,比如一个表单的例子,表单的字段不可能改变,因为是固定的,每个人都一样,但是表单的值是可以更改的

vue 复制代码
<template>
  <div>
    <form @submit.prevent="submitForm">
      <input v-model="formData.name" placeholder="姓名" />
      <input v-model="formData.email" placeholder="邮箱" />
      <button type="submit">提交</button>
    </form>
    <button @click="resetForm">重置表单</button>
  </div>
</template>

<script setup>
import { reactive, shallowReadonly } from 'vue'

// 表单数据结构(可修改)
const formData = reactive({
  name: '',
  email: '',
  // 假设有大量其他字段
})

// 对外暴露只读的结构(防止外部意外替换整个对象)
// 但内部字段值仍可修改
const readonlyForm = shallowReadonly(formData)

// 外部组件如果尝试修改顶层属性会失败,但修改 name/email 仍有效
function resetForm() {
  // 尝试直接替换对象(会失败)
  // readonlyForm = { name: '', email: '' }  // ❌ 不允许
  // 正确做法:通过原对象修改
  formData.name = ''
  formData.email = ''
}

function submitForm() {
  // 提交时使用 readonlyForm 或者 formData 都可以
  console.log('提交数据:', readonlyForm)
}
</script>
相关推荐
大家的林语冰4 小时前
ES5 凉凉,Babel 8 正式发布,默认不再编译为 ES5 和 CJS......
前端·javascript·前端工程化
weedsfly6 小时前
异步编程全景与事件循环——彻底搞懂 JS 执行机制
前端·javascript
用户1733598075376 小时前
纯前端 PDF 数字签名实战:Vue 3 + pdf-lib 在浏览器里完成签名嵌入
前端·javascript
JieE21217 小时前
LeetCode 226. 翻转二叉树|JS 递归超详细拆解,二叉树入门经典题
javascript·算法
JieE21217 小时前
LeetCode 104. 二叉树的最大深度|递归思路超详细拆解
javascript·算法
kyriewen21 小时前
我用 AI 一周写完了整个项目,上线第一天就崩了——这是我踩过最贵的 5 个坑
前端·javascript·ai编程
Larcher21 小时前
AI Loop:让AI像人一样自主完成任务的核心机制
javascript·人工智能·设计模式
默_笙21 小时前
🃏 JS 只有 8 种数据类型,但我花了 2 天才搞懂 null 和 undefined 的区别
javascript
jump_jump1 天前
流式 HTML:从 htmx 片段装配到浏览器原生增量渲染
javascript·性能优化·前端工程化
swipe1 天前
正则表达式入门到进阶:从表单校验到手写模板引擎
前端·javascript·面试