一文了解——to系全家桶并带讲解源码(Vue.js)

To系全家桶并带源码

1.toRef

1.1toRef的例子

举一个例子

vue-sfc 复制代码
<template>
  <div>
     <div>{{ who}}</div>
  </div>
  <hr>
  <button @click="change">修改</button>
</template>

<script setup lang="ts">
const who = {
  name:'kun',
  age:'2.5',
  like:'sing',
}
const change = () => {
  who.like='jump'
  console.log(who.like)
}
</script>

在这个例子中打印出来的是 jump

而在其中的视图 {{who}} 的打印值不会改变 因为其不是响应式的 展示如下

而在此时我们需要加入这小节的核心 也就是 toRef

首先我们需要来讲一下 toRef 的参数 第一个传入的是 对象 第二个传入的是 那个对象的key

在这一点 可以具体的看 vue官网的 类型示例

了解了这个之后 我们改变在上面的例子 加入了 toRef

vue-sfc 复制代码
<template>
  <div>
     <div>{{ who}}</div>
  </div>
  <hr>
  <button @click="change">修改</button>
</template>

<script setup lang="ts">
import {toRef} from 'vue'
const who = {
  name:'kun',
  age:'2.5',
  like:'sing',
}
const like = toRef(who,'like') 

const change = () => {
  like.value = 'jump'
  console.log(like)
  console.log(who)
}
</script>

打印出来的值 分别是

可以看到 其中的数值发生了改变 但是 视图依旧没有发生变化

接下来 我们需要将who这个对象加入响应式

vue-sfc 复制代码
<template>
  <div>
     <div>{{ who}}</div>
  </div>
  <hr>
  <button @click="change">+</button>
</template>

<script setup lang="ts">
import {toRef,reactive} from 'vue'

const who = reactive({
  name:'kun',
  age:'2.5',
  like:'sing',
})

const like = toRef(who,'like') 

const change = () => {
  like.value = 'jump'
  console.log(like)
  console.log(who)
}


</script>

在这里打印出来的是

并且可以看到 视图也同时发生了变化

直白点说就是 toRef 假设非响应式对象 它可以将对象中的值 变成响应式 也可以修改 但是修改不会影响视图的效果

toRef 假设为响应式对象 它可以将对象中的值修改 且会影响视图的效果

1.2toRef的源码

路径如下

在这里映入眼帘的是一个重载函数

重点其实在这里

在这里可以看到 他两个参数都放在这里 并且做了一个判断是否为ref对象

如果不是的话 就会进入到 ObjectRefImpl

这里面 如果看过ref的源码就会发现 很是相像

但是又有不一样的地方 在这里和ref对象不一样的地方就在于 他没有做收集依赖 也没有触发依赖的操作 所以说他对于非响应式对象是不会去改变视图的

2.toRefs

2.1toRefs的例子

单看名字就可以了解到多了一个s 简单的来说就是它可以将对象内的参数 多个传入 并且将其变成响应式的过程

vue-sfc 复制代码
<template>
  <div>
     <div>{{ who}}</div>
  </div>
  <hr>
  <button @click="change">+</button>
</template>

<script setup lang="ts">
import {toRef,reactive,toRefs} from 'vue'

const who = reactive({
  name:'kun',
  age:'2.5',
  like:'sing',
})

const like = toRefs(who) 

const change = () => {
  console.log(like)
}

</script>

在这里直接上图

懂得了他的原理后 我们甚至可以手写他的过程 基本可以做到和源码大差不差

2.2手写toRefs的源码

vue-sfc 复制代码
<template>
  <div>
     <div>{{ who}}</div>
  </div>
  <hr>
  <button @click="change">+</button>
</template>

<script setup lang="ts">
import {toRef,reactive,toRaw} from 'vue'

const who = reactive({
  name:'kun',
  age:'2.5',
  like:'sing',
})

const toRefs =<T extends object> (object:T)=>{
  const result = {} as any
  for (let key in object) {
    result[key] = toRef(object,key)
  }
  return result
}

const change = () => {
  console.log(toRefs(who))
}

</script>

上图

在这里获得效果基本与直接引用效果相同

toRefs的隐藏问题

假使 你想要直接解构它 通过

vue-sfc 复制代码
<template>
  <div>
     <div>{{ who}}</div>
  </div>
  <hr>
  <button @click="change">+</button>
</template>

<script setup lang="ts">
import {toRef,reactive,toRaw} from 'vue'

const who = reactive({
  name:'kun',
  age:'2.5',
  like:'sing',
})

const toRefs =<T extends object> (object:T)=>{
  const result = {} as any
  for (let key in object) {
    result[key] = toRef(object,key)
  }
  return result
}
const {name,age,like} = who
const change = () => {
   who.name = '坤' 
  console.log(name,age,like)
}

</script>

打印出来的结果 值会变化 但是视图依旧是不会的

所以你需要在解构的时候加上 toRefs

const {name,age,like} = who

就可以解决这个问题啦

toRefs的源码

路径也是在ref.ts里面

基本和上面手写的一致

toRaw

toRaw的例子

这个首先我们不讲解 先通过代码先来了解一下这个是干什么的

vue-sfc 复制代码
<template>
  <div>
     <div>{{ who}}</div>
  </div>
  <hr>
  <button @click="change">+</button>
</template>

<script setup lang="ts">
import {reactive,toRaw} from 'vue'

const who = reactive({
  name:'kun',
  age:'2.5',
  like:'sing',
})

const change = () => {
  console.log(who,toRaw(who))
}
</script>

打印出来如下

是不是很好理解 它将其的响应式干掉了 脱离了proxy

通过这样子的操作就可以让他不更新视图的展示了

toRaw手写源码

其实也就是一行的事情 大家可以去试试

vue-sfc 复制代码
<template>
  <div>
     <div>{{ who}}</div>
  </div>
  <hr>
  <button @click="change">+</button>
</template>

<script setup lang="ts">
import {reactive,toRaw} from 'vue'

const who = reactive({
  name:'kun',
  age:'2.5',
  like:'sing',
})

const change = () => {
  console.log(who,who['__v_raw'])
}
</script>

这个方法是看不到的 因为vue没有暴露出来

toRaw的源码

路径在reactive.ts中

就只有一小片

可以看到 他从对象中去获取了一个属性

点过去就可以看见这些了

相关推荐
程序员码歌17 分钟前
短思考第261天,浪费时间的十个低效行为,看看你中了几个?
前端·ai编程
Swift社区1 小时前
React Navigation 生命周期完整心智模型
前端·react.js·前端框架
若梦plus1 小时前
从微信公众号&小程序的SDK剖析JSBridge
前端
用泥种荷花2 小时前
Python环境安装
前端
Light602 小时前
性能提升 60%:前端性能优化终极指南
前端·性能优化·图片压缩·渲染优化·按需拆包·边缘缓存·ai 自动化
Jimmy2 小时前
年终总结 - 2025 故事集
前端·后端·程序员
烛阴2 小时前
C# 正则表达式(2):Regex 基础语法与常用 API 全解析
前端·正则表达式·c#
roman_日积跬步-终至千里2 小时前
【人工智能导论】02-搜索-高级搜索策略探索篇:从约束满足到博弈搜索
java·前端·人工智能
GIS之路2 小时前
GIS 数据转换:使用 GDAL 将 TXT 转换为 Shp 数据
前端
多看书少吃饭2 小时前
从Vue到Nuxt.js
前端·javascript·vue.js