Vue2 ➔ Vue3 都做了哪些改变?

news2024/11/27 12:47:57

Vue 2 vs. Vue 3: What Are the Differences and Which Version Should You  Choose?

不是吧,兄弟,Vue3 都出来多久了,你还对这个感兴趣,说!是不是没好好卷?😏

俺也一样 😂,Vue3 出来之后只是简单了解了一下,然后还是转头一直在写 Vue2。当然,这也和大家搬砖 🧱 的处境有关。一般情况下,用 Vue2 起的项目没有什么大的问题,谁又会耗费经历去迁移呢?

不过自己在有机会单独写 H5 的时候,还是会有意识的去试试 Vue3 的。感觉 Vue2 如果掌握的不错的话,其实学习 Vue3 也没什么大的成本,但如果突然让你总结一下 Vue2 和 Vue3 的区别,你能答上来多少呢?数据响应式?生命周期?阿巴阿巴... 😅

其实官方 Vue3 迁移指南 早就给出了详细的说明,这里对一些常用的做下些总结,一起看看吧,Let's go!🚀

1. 语法上

1.1 指令模板相关

  • 移除 v-on.native 修饰符

在 Vue 2 中,v-on.native 修饰符使得组件的事件侦听器可以直接绑定到组件根元素上的原生事件。至于为什么要在 Vue3 中移除,网上有这样的解释:Vue2 这种方式并不符合组件的封装和规范性,因为组件应该将自己的行为和内部实现细节封装起来,而不是直接依赖于父级组件的事件。

Vue 3 更加倡导组件的封装性和独立性,因此移除了v-on.native 修饰符,鼓励使用更明确的方式进行事件传递和处理。有道理 🤔

  • v-model 语法调整,v-model:title="title",代替了之前的 v-model 和 .sync
<!-- Vue2: -->
<ChildComponent v-model="pageTitle" />
<!-- 是以下的简写: -->
<ChildComponent :value="pageTitle" @input="pageTitle = $event" />

<ChildComponent :title.sync="pageTitle" />
<!-- 是以下的简写: -->
<ChildComponent :title="pageTitle" @update:title="pageTitle = $event" />


<!-- Vue3: -->
<ChildComponent v-model:title="pageTitle" />
<!-- 是以下的简写: -->
<ChildComponent :title="pageTitle" @update:title="pageTitle = $event" />

1.2 组件相关

  • 异步组件的调整,路由懒加载的组件需要用 defineAsyncComponent 关键字包裹 📦
  • emit 选项可以定义组件可向父组件触发的事件,同 props
// 数组语法
export default {
  emits: ['check'],
  created() {
    this.$emit('check')
  }
}

// 对象语法
export default {
  emits: {
    // 没有验证函数
    click: null,

    // 具有验证函数
    submit: (payload) => {
      if (payload.email && payload.password) {
        return true
      } else {
        console.warn(`Invalid submit event payload!`)
        return false
      }
    }
  }
}

1.3 渲染函数

  • $listeners 合并至 $attrs
  • $attrs 现在包含了所有传递给组件的 attribute,包括 class 和 style

1.4 移除的 API

  • createApp() 代替 new Vue()
  • ​从实例中完全移除了 $on、$off 和 $once 方法,通过 new Vue() 方式实现的事件总线不再可用 😅。官方推荐用外部的、实现了事件触发器接口的库, 如 mitt 或 tiny-emitter

1.5 其他

  • 根元素可以有不止一个标签了
  • create 生命周期被 setup 代替,destroyed 被改名为 unmounted
  • data 选项已标准化为只接受返回 object 的 function
  • watch 可以侦听数组时候,必须指定 deep 才能侦听到数组元素的改变
  • 新增了 <Teleport> 传送门
  • ref 支持函数了

2. 数据响应式

2.1 Vue2 数据响应式原理

老生常谈的问题,再赘述一遍 😐

Vue2 对数据的劫持是用的 Object.defineProperty,但这种方式其实是有缺点的 👀

  • 对于未初始化但却直接在 DOM 上引用的变量,并不能做到响应式 ➡️ 补救:$set
  • 对于数组类型的数据,通过下标改变的方式达不到响应式 ➡️ 补救:使用改写的数组 API
Object.defineProperty(obj, key, val) {
	enumerable: true,
	configurable: true,
	get: function reactiveGetter() {
		return val
	},
	set: function reactiveSetter(newVal) {
		if(newVal === val) return
		val = newVal
		dep.notify() // 通知依赖该属性的其他组件进行更新
	}
}

2.2 Vue3 数据响应式原理

​Vue3 弥补了 Object.defineProperty 的两个不足:

  • Vue2 中动态创建的 data 属性需要通过 Vue.$set 来赋,Proxy 则不需要
  • 基于性能的考虑,Vue2 篡改了数组的 7 个 API,Proxy 则不需要

defineProperty 需要提前递归遍历 data 做到响应式,而 Proxy可以在真正用到深层数据时候再做响应式(惰性)🐂🍺

funciton reactive(obj) {
	return new Proxy(obj, {
		get(target, key, receiver) {
			const res = Reflect.get(target, key, receiver)
			track(target, key) // 收集依赖
			return isObject(res) ? reactive(res) : res
		},
		set(target, key, value, receiver) {
			const oldValue = target[key]
			const result = Reflect.set(target, key, value, receiver)
			if(result && oldValue !== value) {
				trigger(target, key) // 触发更新
			}
			return result
		}
	})
}

3. Composition API

 Vue 官方:组合式 API 常见问答:组合式 API (Composition API) 是一系列 API 的集合,使我们可以使用函数而不是声明选项的方式书写 Vue 组件。它是一个概括性的术语,涵盖了以下方面的 API:

在 Vue 3 中,组合式 API 基本上都会配合 <script setup> 语法在单文件组件中使用。下面是一个使用组合式 API 的组件示例:

<template>
  <button @click="increment">点击了:{{ count }} 次</button>
</template>

<script setup>
import { ref, onMounted } from 'vue'

// 响应式状态
const count = ref(0)

// 更改状态、触发更新的函数
function increment() {
  count.value++
}

// 生命周期钩子
onMounted(() => {
  console.log(`计数器初始值为 ${count.value}。`)
})
</script>

为什么要用 Composition API?

Vue 官方:组合式 API 常见问答 里也说了为什么,比如更好的逻辑复用、类型推导、更灵活的代码组织、更小的生产包体积。🐂🍺

此外,我还了解大概是 2019 年 6月,尤雨溪发表了一篇 Composition API 的文章:「Vue Function-based API RFC」 ,详细阐述了为什么要出新的 API,其描述的设计思路大致总结如下:

  • 响应式 API:例如 ref() 和 reactive(),使我们可以直接创建响应式状态、计算属性和侦听器。
  • 生命周期钩子🪝:例如 onMounted() 和 onUnmounted(),使我们可以在组件各个生命周期阶段添加逻辑。
  • 依赖注入:例如 provide() 和 inject(),使我们可以在使用响应式 API 时,利用 Vue 的依赖注入系统。
  • Composition API 比 mixins、高阶组件、Renderless Components 更好:
    • 模版中的数据来源不清晰。举例来说,当一个组件中使用了多个 mixin 的时候,光看模版会很难分清一个属性到底是来自哪一个 mixin。HOC 也有类似的问题。
    • 命名空间冲突。由不同开发者开发的 mixin 无法保证不会正好用到一样的属性或是方法名。HOC 在注入的 props 中也存在类似问题。
    • 性能。HOC 和 Renderless Components 都需要额外的组件实例嵌套来封装逻辑,导致无谓的性能开销。
  • Composition API 更适合 TypeScript,Vue2 的装饰器结合 TypeScript 非常难用。

ps:不知道大家有没有关注过 Vue3 的整个推出过程,我记得 Vue3 一开始公布的想法是推出一个基于 Class 的 API,忘记是在哪里偶然看到的了。那个时候 React 推出了 hooks 的概念,不知道是不是觉的 hooks 的概念更好,但是又不能直接叫 hooks API,所以就叫 Composition API。

此外,当时还有一个潮流,大家好像都在从 JS 往 TS 上靠,Vue 应该也想和 TypeScript 结合的更紧密,单 Vue2 装饰器的写法非常麻烦,于是乎一石二鸟,Composition API 即赶上了最新的 API 风格,又赶上了最新的语言风格。ps:也许是我凭空 yy 🤔

​从个人的角度来讲,作为 Vue 的使用者,我感觉 Vue3 比 React 要好。因为 Vue3 是在 React 后推出的,它看到 hooks API 之后才发明的 Composition API,那它就避免了 React 中的几个显著的问题,比如:不能获取最新的值,手动写依赖不太容易写好,可能某个依赖忘写导致不更新...

end 🎉

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/764247.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

如何升级iOS17/iPadOS17公测版?iOS17公测版升级教程

苹果官方发布了iOS 17/iPadOS 17系统首个公测版更新&#xff0c;其版本号及更新内容与iOS 17 beta 3一致&#xff0c;版本号为21A5277j。 对于想升级iOS 17/iPadOS 17 公测版的用户&#xff0c;可以参考本教程进行操作。 升级注意事项&#xff1a; 1. 为防止意外情况&#xf…

AlignPS中的TOIM损失

本文介绍了CVPR2021行人重识别领域中一篇名为AlignPS论文中的TOIM损失函数 论文链接&#xff1a;https://arxiv.org/abs/2109.00211 代码链接&#xff1a;GitHub - daodaofr/AlignPS: Code for CVPR 2021 paper: Anchor-Free Person Search TOIM TOIM Loss OIM Loss Trip…

【STM32零基础入门教程02】STM32环境获取及搭建

(14条消息) 【STM32零基础入门教程01】STM32入门基础知识_小超电子笔记的博客-CSDN博客 工欲善其事必先利其器&#xff0c;在上一章了解完STM32的一些基础知识之后&#xff0c;我们需要对使用的环境进行获取和安装。 一、MDK&#xff08;KEIL&#xff09;的获取 Keil是一种…

CCF 202209-2 何以包邮? (01背包动态规划练习)

一、先温习一下01背包问题 有N件物品和一个容量为V的背包。第i件物品的体积是c[i]&#xff0c;价值是w[i]。求解将哪些物品装入背包可使价值总和最大。 条件汇总 -------- 背包限制容量&#xff1a;Z 此时背包容量&#xff1a;C 物品&#xff1a;1 , i ... …

WSA - root,frida与ida测试

本文旨在配置windows subsystem for android(win安卓子系统)来作为win在开启了hyper-v的情况下的一种轻量的安卓模拟器方案。使用MagiskOnWsa设置root权限&#xff0c;最终使其正常与开发环境、frida、ida打通。 1. Root的WSA 常用的Wsa版本在目前是没有默认root的。在物理机…

matlab滤波器设计-IIR滤波器的设计与仿真

matlab滤波器设计-IIR滤波器的设计与仿真 1 引言 在现代通信系统中&#xff0c;由于信号中经常混有各种复杂成分&#xff0c;所以很多信号的处理和分析都是基于滤波器而进行的。但是&#xff0c;传统的数字滤波器的设计使用繁琐的公式计算&#xff0c;改变参数后需要重新计…

【Matlab】智能优化算法_猎豹优化算法CO)

【Matlab】智能优化算法_猎豹优化算法CO 1.背景介绍2.数学模型2.1 搜索策略2.2 坐等策略2.3 攻击策略2.4 假设 3.文件结构4.伪代码5.详细代码及注释5.1 CO.m5.2 CO_VectorBased.m5.3 Get_Functions_details.m 6.运行结果7.参考文献 1.背景介绍 猎豹&#xff08;Achinonyx jubat…

Kubernetes部署服务到集群中的指定节点

# kubectl get node NAME STATUS ROLES AGE VERSION k8s-master Ready master 25h v1.17.3 k8s-node2 Ready <none> 25h v1.17.3 集群只有两个节点&#xff0c;这里打算将应用部署在k8s-node2节点上&#xff0c;需要先记下这个节点的…

【3】Vite Vue3 用户、角色、岗位选择组件封装

在当今前端开发的领域里&#xff0c;快速、高效的项目构建工具以及使用最新技术栈是非常关键的。ViteVue3 组合为一体的项目实战示例专栏将带领你深入了解和掌握这一最新的前端开发工具和框架。 作为下一代前端构建工具&#xff0c;Vite 在开发中的启动速度和热重载方面具有突…

攻不下dfs不参加比赛(十三)

标题 为什么练dfs题目为什么练dfs 相信学过数据结构的朋友都知道dfs(深度优先搜索)是里面相当重要的一种搜索算法,可能直接说大家感受不到有条件的大家可以去看看一些算法比赛。这些比赛中每一届或多或少都会牵扯到dfs,可能提到dfs大家都知道但是我们为了避免眼高手低有的东…

24 - 数组和广义表 - 二维数组

前面我们学习了一维数组、今天来看看二维数组,比一维数组更加复杂! 数组的特点 存储的空间连续 存储类型相同 可以使用地址+偏移快速访问 二维数组定义 二维数组本质上是以数组作为数组元素的数组,即“数组的数组”,语法格式如下: 类型说明符 数组名[常量表达式][常量表达…

977.有序数组的平方

977.有序数组的平方 1.暴力排序 这道题最直观的方法在于&#xff0c;将数组中的每个数平方之后&#xff0c;排个序 public int[] sortedSquares(int[] nums) {int[]ans new int[nums.length];for(int i0;i<nums.length;i){ans[i] nums[i]*nums[i];} Arrays.sort(ans);ret…

人工智能LLM模型:奖励模型的训练、PPO 强化学习的训练、RLHF

人工智能LLM模型&#xff1a;奖励模型的训练、PPO 强化学习的训练 1.奖励模型的训练 1.1大语言模型中奖励模型的概念 在大语言模型完成 SFT 监督微调后&#xff0c;下一阶段是构建一个奖励模型来对问答对作出得分评价。奖励模型源于强化学习中的奖励函数&#xff0c;能对当前…

高通芯片android进入EDL模式 下载 热启动 串口指令

参考&#xff1a;高通方案的Android设备几种开机模式的进入与退出_edl模式怎么退出_Rookie20190715的博客-CSDN博客 切换为EDL模式 向串口发送 4b 65 01 00 54 0f 7e 或者adb reboot edl

Ceph的安装部署

文章目录 一、存储基础1.1 单机存储设备1.2 单机存储的问题1.3分布式存储&#xff08;软件定义的存储 SDS&#xff09; 二、Ceph 简介2.1 Ceph 优势2.2 Ceph 架构2.3 Ceph 核心组件2.4 Pool、PG 和 OSD 的关系&#xff1a;2.5 OSD 存储后端2.6 Ceph 数据的存储过程2.7 Ceph 版本…

PID控制系列--(1、最形象的PID)

目录 1、 比例控制系统的标准结构2、最简单的例子3、第二个例子4、积分控制器6、微分控制7 总结 今天 看到了B站上一个叫洋葱auto的UP主搬来的介绍PID控制的视频&#xff0c;感觉讲得形象易懂&#xff0c;为便于让和我一样看了无数文章还是不能很好理解PID控制本质的人共同分享…

2. DATASETS DATALOADERS

2. DATASETS & DATALOADERS PyTorch提供了两个数据基元&#xff1a;torch.utils.data.DataLoader和torch.uutils.data.data集&#xff0c;允许使用预加载的数据集以及自己的数据。数据集存储样本及其相应的标签&#xff0c;DataLoader在数据集周围包装了一个可迭代项&…

Sentinel整合OpenFegin

之前学习了openFeign的使用&#xff0c;我是超链接 现在学习通过Sentinel来进行整合OpenFegin。 引入OpenFegin 我们需要在当前的8084项目中引入对应的依赖 <dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-sta…

网络套接字编程(一)(UDP)

gitee仓库&#xff1a;https://gitee.com/WangZihao64/linux/tree/master/chat_udp 预备知识 源IP地址和目的IP地址 它是用来标识网络中不同主机的地址。两台主机进行通信时&#xff0c;发送方需要知道自己往哪一台主机发送&#xff0c;这就需要知道接受方主机的的IP地址&am…

【数学建模】利用C语言来实现 太阳赤纬 太阳高度角 太阳方位角 计算和求解分析 树木树冠阴影面积与种植间距的编程计算分析研究

太阳赤纬的计算 #include <stdio.h> #include <math.h>double calculateDelta(int year, int month, int day, int hour, int minute, int second) {int n, n0;double t, theta, delta;// 计算n和n0n month * 30 day;n0 79.6764 0.2422 * (year - 1985) - ((y…