Vue3-原始值的响应式方案ref

一、原始类型的值

原始类型的指的是: booleannumberstringsymbolundefindnull等类型的值.

一、初识ref

为什么vue3需要对原始值的响应式做单独处理?因为Javascript中的Proxy只能代理对象类型的数据, 如普通对象、数组、Set、Map等。 为了解决Proxy不能代理原始类型的问题,vue3使用对象类型包裹原始类型的方法,也就是提供一个ref函数,如下是ref函数的简单实现:

javascript 复制代码
function ref(val) {
  const wrapper = {
    value: val
  }
  return reactive(wrapper)
}

const count = ref(1)
count++

这就是使用ref创建的响应式数据时要加.vlaue访问的原因,因为在ref函数中使用了对象类型 包裹 原始类型,然后使用reactive将包裹对象变成响应式数据并返回。实际上vue3源码对ref的实现做了很多抽象和封装并对不同类型的数据做了严格的逻辑判断,想要更深一步的了解ref可自行阅读vue3 ref 相关源码,在此就不过多赘述了。

二、模板中自动脱ref

为了减轻开发者的心智负担,vue3在模板访问和设置ref时实现了自动脱ref,自动脱ref需要对包裹对象 添加唯一标识,这个唯一标识就是在包裹对象 上添加一个__v_isRef只读属性,它用来判断一个对象是否为ref。我们需要修改一下ref函数的实现:

javascript 复制代码
function ref(val) {
  const wrapper = {
    value: val
  }
  Object.definePrototype(wrapper, '__v_isRef',{
    value: ture
  })
  return reactive(wrapper)
}

Object.definePrototype为对象配置属性通常会用到三个描述符

  1. configurable 默认值为false 也就是不可配置
  2. writable 默认值为false 也就是不可写
  3. enumerable 默认值为false 也就是不可枚举 ,即for...in访问不到该属性

自动脱ref函数的简单实现

javascript 复制代码
function proxyRefs(objectWithRef) {
  return new Proxy(objectWithRef, {
  	get(target, key) {
      return unref(target)
    },
  	set(target, key, value, receiver) {
  	  const oldValue = target[key]
  	  if (isRef(oldValue) && !isRef(value)) {
	    oldValue.value = value
	    return true
	  } else {
	    return Reflect.set(target, key, value, receiver)
	  }
  	}
  })
}

function unref(ref) {
  return isRef(ref) ? ref.value : ref
}

function isRef(r) {
  return !!(r && r.__v_isRef === true)
}

三、解决响应式丢失问题

ref除用来为原始值提供响应式方案,还可以解决响应式丢失的问题,比如下面的一个例子:

javascript 复制代码
<template>
  <div>{{ count }}</div>
</template>

<script>
//
import { reactive } from 'vue'

export default {
  setup() {
    const proxy = reactive({ count: 1 })

    setTimeout(() => {
      proxy.count++
    }, 3000)

    return {
      ...proxy
    }
  }
}
</script>

运行上述代码,定时器3s后会修改响应式数据的值但模板中的数据并未更新,其根本原因出在...操作符上

javascript 复制代码
return {
  ...proxy
}

等价于:

javascript 复制代码
return {
  count: 1
}

返回的是一个普通对象,所以在模板中读取该对象的属性时不会与渲染函数建立联系,修改proxy的属性时自然也不会触发渲染函数的再次执行。故此toReftoRefs函数就应运而生,下面是toRef和toRefs函数的简单实现:

javascript 复制代码
function toRef(target, key) {
  const wrapper = {
    get value() {
	  return target[key]
	},
	set value(value) {
	  target[key] = value
	}
  }
  Object.definePrototype(wrapper, '__v_isRef',{
    value: ture
  })
  return wrapper
}

function toRefs(target) {
  const result = {}
  for (const key in target) {
    result[key] = toRef(target, key)	
  }
  return result
}

故此我们只要修改script中的一行代码就可以避免响应式的丢失了

javascript 复制代码
return {
  ...toRefs(proxy)
}
相关推荐
啦啦9118869 小时前
【版本更新】Edge 浏览器 v142.0.3595.94 绿色增强版+官方安装包
前端·edge
蚂蚁集团数据体验技术10 小时前
一个可以补充 Mermaid 的可视化组件库 Infographic
前端·javascript·llm
LQW_home10 小时前
前端展示 接受springboot Flux数据demo
前端·css·css3
q***d17310 小时前
前端增强现实案例
前端·ar
IT_陈寒10 小时前
Vite 3.0 重磅升级:5个你必须掌握的优化技巧和实战应用
前端·人工智能·后端
JarvanMo10 小时前
Flutter 3.38 + Firebase:2025 年开发者必看的新变化
前端
Lethehong10 小时前
简历优化大师:基于React与AI技术的智能简历优化系统开发实践
前端·人工智能·react.js·kimi k2·蓝耘元生代·蓝耘maas
华仔啊10 小时前
还在用 WebSocket 做实时通信?SSE 可能更简单
前端·javascript
鹏北海11 小时前
多标签页登录状态同步:一个简单而有效的解决方案
前端·面试·架构
_AaronWong11 小时前
基于 Vue 3 的屏幕音频捕获实现:从原理到实践
前端·vue.js·音视频开发