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>
相关推荐
yqcoder4 小时前
JavaScript 浅拷贝:只复制“第一层”的艺术
开发语言·javascript·ecmascript
yqcoder4 小时前
JavaScript 闭包:函数背后的“背包”
开发语言·javascript·ecmascript
threelab4 小时前
挑战AI辅助从零构建3D模型编辑器:01基于Vue3 + Three.js的现代化架构设计
javascript·人工智能·3d·前端框架·着色器
invicinble4 小时前
前端框架使用vue-cli (第五层:构建打包层--babel.config.js介绍)
前端·vue.js·前端框架
前端若水4 小时前
安装 markdown-it 后项目报错,可能是 Vue/Webpack 项目中 Quill 的问题(ES6+ 语法不支持)
vue.js·webpack·es6
张元清4 小时前
React 浏览器标签页 UX:用标题、Favicon 和通知把用户拉回来
前端·javascript·面试
葛兰岱尔4 小时前
葛兰岱尔rapid3D Loader for Three.js使用方式及7个基础API说明
开发语言·javascript·3d
Lkstar4 小时前
读完红宝书和YDKJS,我终于搞懂了原型链、闭包和this
javascript·面试
用户11489669441054 小时前
JavaScript原型链解析
javascript