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>
相关推荐
LXXgalaxy3 小时前
Uni-app 小程序页面跳转带参实战笔记(含对象传参与防坑)
开发语言·前端·javascript
2301_768350233 小时前
Vue指令修饰符
前端·javascript·vue.js
oi..3 小时前
Flag和JavaScript document有关
开发语言·前端·javascript·经验分享·笔记·安全·网络安全
每天吃饭的羊3 小时前
computed 同时写 get() 和 set()
前端·javascript·vue.js
Highcharts.js3 小时前
Highcharts + TypeScript 集成高级技巧|类型与框架集成实战
前端·javascript·vue.js·react.js·typescript·highcharts·图表生成
luanma1509803 小时前
Vue2 vs Vue3:核心区别全解析
前端·javascript·vue.js
qq_381338503 小时前
Vue 3 组合式 API 最佳实践:从入门到精通
前端·javascript·vue.js
小王码农记3 小时前
el-input限制只能输入价格格式
前端·vue.js
李剑一3 小时前
纯干货,前端字体极致优化!谷歌、阿里、字节、腾讯都在用的终极解决方案,Vue3 + Vite 直接抄,页面提速不妥协!
前端·vue.js·面试