当前位置: 首页 > article >正文

Vue3的Composition组合式API(readonly与shallowReadonly函数、toRaw与markRaw函数、customRef函数)

目录

  • 1. readonly与shallowReadonly函数
  • 3. toRaw与markRaw函数
  • 4. customRef函数

1. readonly与shallowReadonly函数

  • readonly: 让一个响应式数据变为只读的(深只读)
  • shallowReadonly:让一个响应式数据变为只读的(浅只读),可以修改第二层及更深的层数据
  • 应用场景: 不希望数据被修改时。如别人提供的数据,我们只想使用,不想修改

使用示例

<template>
  <h2>当前x的值是:{{x}}</h2>
  <button @click="x++">点我x++</button>

  <h2>姓名:{{person.name}}</h2>
  <h2>薪资:{{person.job.salary}}K</h2>
  <button @click="person.name+='~'">修改姓名</button>
  <button @click="person.job.salary++">涨薪</button>
</template>

<script>
import {ref,reactive,readonly,shallowReadonly} from 'vue'
export default {
  name: 'Student',
  setup(){
    let x = ref(0)

    let person = reactive({
      name:'张三',
      job:{
        salary:20
      }
    })

    x = readonly(x)
    person = shallowReadonly(person)

    return {
      x,
      person
    }
  }
}
</script>

效果如下:

  1. 点我x++,页面没反应
  2. 点击修改姓名,页面没反应
  3. 点击涨薪,页面上的薪资能正常变化
    readonly与shallowReadonly函数

3. toRaw与markRaw函数

toRaw:

  • 作用:将一个由reactive生成的响应式对象转为普通对象
  • 注意:对这个普通对象的所有操作,会引起响应式对象的数据发生变化,但不会引起页面更新。等下一次页面被重新渲染时,发生改变的数据会更新到页面

markRaw:

  • 作用:给一个响应式对象,添加一个对象类型的属性A。标记属性A,使其永远不会再成为响应式对象
  • 注意:对属性A的所有操作,会引起属性A的数据发生变化,但不会引起页面更新。等下一次页面被重新渲染时,属性A发生改变的数据会更新到页面
  • 应用场景:
    1. 有些值不应被设置为响应式的,例如引用的复杂的第三方类库等
    2. 当渲染具有不可变数据源的大列表时,跳过响应式转换可以提高性能

使用示例
Student.vue

<template>
  <h2>人员全部信息: {{person}}</h2>
  <h2>姓名:{{person.name}}</h2>
  <button @click="person.name+='~'">修改姓名</button>

  <br>
  <button @click="getAndChangeRawPerson">先获取再改变最原始的person信息</button>
  <br>
  <button @click="addRawCar">给人添加一台最原始的车</button>
  <br>
  <button v-show="person.car" @click="changeCarPrice">改变车的价格</button>
</template>

<script>
import {reactive,toRaw,markRaw} from 'vue'
export default {
  name: 'Student',
  setup(){

    let person = reactive({
      name:'张三'
    })

    function getAndChangeRawPerson(){
      const rawPerson = toRaw(person)
      // 会引起person的数据变化,但页面不会刷新。页面会在下次渲染时更新改变的数据
      rawPerson.name+='$'
    }

    function addRawCar(){
      let car = {name:'大众',price:10}
      // 如果没有markRaw。person.car = car添加的是一个响应式数据。因为person是响应式的
      person.car = markRaw(car)
    }

    function changeCarPrice() {
      // 会引起person的数据变化,但页面不会刷新。页面会在下次渲染时更新改变的数据
      person.car.price = 99
    }

    return {
      person,
      getAndChangeRawPerson,
      addRawCar,
      changeCarPrice
    }
  }
}
</script>

效果如下:

  1. 点击先获取再改变最原始的person信息,页面没有发生变化,但其实person的数据已经改变了

  2. 再点击修改姓名,页面重新渲染,会将之前对toRaw数据的改变也更新到页面
    toRaw函数效果

  3. 点击给人添加一台最原始的车,页面能正常更新person。同时也会出现改变车的价格按钮
    给人添加一台车的效果

  4. 点击改变车的价格,页面没有发生变化,但其实person.car的数据已经改变了

  5. 再点击修改姓名,页面重新渲染,会将之前对markRaw的person.car的数据的改变也更新到页面
    markRow函数的效果

4. customRef函数

  • 作用:创建一个自定义的ref,并对其读取track跟踪更新trigger触发进行显式控制。
  • 案例:实现防抖效果。如下所示:

App.vue

  • 自定义一个ref。value就是ref管理的数据;delay是其它参数
  • 读取value的值,会自动调用get。需要使用track(),通知Vue追踪value的变化,这样才会加载value更新后的数据
  • 当修改value的值,会自动调用set,变传入需要修改的值。需要使用trigger(),通知Vue去重新解析模板
  • set中创建了一个定时器。延迟一段时间,再去修改数据
  • 使用let keyWord = myRef('hello',500)定义了一个响应式的数据keyWord,初始的数据是hello,每次修改keyword都会延迟500ms
<template>
  <input type="text" v-model="keyWord">
  <h3>{{keyWord}}</h3>
</template>

<script>
import {customRef} from 'vue'
export default {
  name: 'App',
  setup() {
    // 自定义一个ref。value就是ref管理的数据;delay是其它参数
    function myRef(value, delay){
      let timer
      return customRef((track,trigger)=>{
        return {
          // 读取value的值,会自动调用get
          get(){
            console.log(`有人从myRef这个容器中读取数据了,我把${value}给他了`)
            track() // 通知Vue追踪value的变化,这样才会加载value更新后的数据
            return value
          },
          // 当修改value的值,会自动调用set,变传入需要修改的值
          set(newValue){
            console.log(`有人把myRef这个容器中数据改为了:${newValue}`)
            clearTimeout(timer)   // 如果有定时器未执行,则先清除旧的定时器,防止抖动
            timer = setTimeout(()=>{    // 延迟一段时间,再去修改数据
              value = newValue
              trigger() // 通知Vue去重新解析模板
            },delay)
          }
        }
      })
    }

    // 使用程自定义的ref
    let keyWord = myRef('hello',500)

    return {keyWord}
  }
}
</script>

效果如下:在输入框的hello后面,快速的输入123,过一段时间,下面的hello也更新为hello123
customRef函数


http://www.kler.cn/news/367106.html

相关文章:

  • 基于物联网的智慧考场系统设计(论文+源码)
  • PG数据库之事务处理
  • GPS/北斗时空安全隔离装置(卫星时空防护装置)使用手册
  • gin入门教程(5):请求参数处理
  • 【大数据学习 | kafka】kafuka的基础架构
  • 【ShuQiHere】Linux 系统中的硬盘管理详解:命令与技巧
  • [ComfyUI]与 FLUX.1[dev] 一样优秀的商业用途模型 OpenFLUX.1 现已面世!
  • 常用sql
  • 记录下docker部署gitlab-ce-17.5版本及客户端git拉取方式配置
  • AI视频!OpenAI发布最新模型sCM,开启图像、音频、视频、三维模型AI新时代
  • 【C++进阶】深入STL之list:模拟实现深入理解List与迭代器
  • Vscode + EIDE +CortexDebug 调试Stm32(记录)
  • ATom:2016-2018 年沿飞行轨迹的 CAM-chem/CESM2 模型输出
  • 编写一个简单的Iinput_dev框架
  • 权益资本成本-CAPM模型、MPEG模型、OJ模型、PEG模型、原始数据及其代码(2000-2021年)
  • 【ESP32S3 Sense接入阿里云大模型图像理解】
  • GDB 从裸奔到穿戴整齐
  • 2024 BuildCTF 公开赛|Crypto
  • SpringBoot中EasyExcel使用实践总结
  • 【Redis 设计与实现】String 的数据结构如何实现的?
  • RN安卓应用打包指南
  • 帝国CMS 内容页调用上一篇下一篇的方法(精华汇总)
  • 零一万物新模型Yi-Lightning:超越GPT-4o
  • C#实现简单的文件夹对比程序(续)
  • 《使用Gin框架构建分布式应用》阅读笔记:p208-p211
  • 函数连续性导论