一个 Java 猿眼中 Vue3 和 Vue2 的差异
ztj100 2025-06-10 04:17 11 浏览 0 评论
随着 TienChin 项目视频的录制,松哥终于也要静下心来,认真捋一捋 Vue3 中的各种新特性了,然后再和小伙伴们进行分享,其实 Vue3 中还是带来了很多新鲜的玩意,今天我们就不卷 Java 了,来卷卷前端。
以下内容是一个 Java 猿对 Vue3 的理解,主要是应用层面上,如果有专业的前端小伙伴,请轻拍。
1. script 写法
进入到 Vue3 时代,最明显的感受就是在一个 .vue 文件中,script 标签的写法大变样了。以前在 Vue2 中,我们都是这样写的:
<script>
export default {
name: "SysHr",
data() {
return {
//
}
},
mounted() {
//
},
methods: {
deleteHr(hr) {
//
},
doSearch() {
//
}
}
}
</script>
不过到了 Vue3 里边,这个写法变了,变成下面这样了:
<template>
<div>
<div>{{a}}</div>
<div>{{result}}</div>
<button @click="btnClick">clickMe</button>
</div>
</template>
<script>
import {ref} from 'vue';
import {onMounted,computed} from 'vue'
export default {
name: "MyVue01",
setup() {
const a = ref(1);
const btnClick=()=>{
a.value++;
}
onMounted(() => {
a.value++;
});
const result = computed(()=>{
return Date.now();
});
return {a,btnClick,result}
}
}
</script>
先从大的方面来看,细节实现咱们后面再细聊。
大的方面,就是在这个 export default 中,以后就只有两个元素了,name 和 setup,我们以前的各种方法定义、生命周期函数、计算属性等等,都写在 setup 中,并且需要在 setup 中返回,setup 中返回了什么,上面的 template 中就能用什么。
这种写法稍微有点费事,所以还有一种简化的写法,像下面这样:
<template>
<div>
<div>{{a}}</div>
<div>{{result}}</div>
<button @click="btnClick">clickMe</button>
</div>
</template>
<script setup>
import {ref} from 'vue';
import {onMounted, computed} from 'vue'
const a = ref(1);
const btnClick = () => {
a.value++;
}
onMounted(() => {
a.value++;
});
const result = computed(() => {
return Date.now();
});
</script>
这种写法就是直接在 script 标签中加入 setup,然后在 script 标签中该怎么定义就怎么定义,也不用 return 了。这个场景,又有点 jQuery 的感觉了。
上面这个实现里有几个细节,我们再来详细说说。
2. 生命周期
首先就是生命周期函数的写法。
以前 Vue2 里的写法有一个专业名词叫做 options API,现在在 Vue3 里也有一个专业名词叫做 composition API。在 Vue3 中,这些对应的生命周期函数都要先从 vue 中导出,然后调用并传入一个回调函数,像我们上一小节那样写。
下图这张表格展示了 options API 和 composition API 的一一对应关系:
options API composition API beforeCreate Not Needed created Not Needed mounted onMounted beforeUpdate onBeforeUpdate updated onUpdated beforeUnmount onBeforeUnmount unmounted onUnmounted errorCaptured onErrorCaptured renderTracked onRenderTracked renderTriggered onRenderTriggered activated onActivated deactivated onDeactivated
想用哪个生命周期函数,就从 vue 中导出这个函数,然后传入回一个回调就可以使用了。例如第一小节中松哥给大家举的 onMounted 的用法。
3. 计算属性
除了生命周期函数,计算属性、watch 监听等等,用法也和生命周期函数类似,需要先从 vue 中导出,导出之后,也是传入一个回调函数就可以使用了。上文有例子,我就不再啰嗦了。
像 watch 的监控,写法如下:
<script>
import {ref} from 'vue';
import {onMounted,computed,watch} from 'vue'
export default {
name: "MyVue01",
setup() {
const a = ref(1);
const btnClick=()=>{
a.value++;
}
onMounted(() => {
a.value++;
});
const result = computed(()=>{
return Date.now();
});
watch(a,(value,oldValue)=>{
console.log("value", value);
console.log("oldValue", oldValue);
})
return {a,btnClick,result}
}
}
</script>
导入 watch 之后,然后直接使用即可。
4. ref 于 reactive
上面的例子中还有一个 ref,这个玩意也需要跟大家介绍下。
在 Vue2 里边,如果我们想要定义响应式数据,一般都是写在 data 函数中的,类似下面这样:
<script>
export default {
name: "SysHr",
data() {
return {
keywords: '',
hrs: [],
selectedRoles: [],
allroles: []
}
}
}
</script>
但是在 Vue3 里边,你已经看不到 data 函数了,那怎么定义响应式数据呢?就是通过 ref 或者 reactive 来定义了。
在第一小节中,我们就是通过 ref 定义了一个名为 a 的响应式变量。
这个 a 在 script 中写的时候,有一个 value 属性,不过在 HTML 中引用的时候,是没有 value 的,可千万别写成了 {{a.value}},我们再来回顾下上文的案例:
<template>
<div>
<div>{{a}}</div>
<button @click="btnClick">clickMe</button>
</div>
</template>
<script>
import {ref} from 'vue';
export default {
name: "MyVue04",
setup() {
const a = ref(1);
const btnClick=()=>{
a.value++;
}
return {a,btnClick}
}
}
</script>
现在就是通过这样的方式来定义响应式对象,修改值的时候,需要用 a.value,但是真正的上面的 template 节点中访问的时候是不需要 value 的(注意,函数也得返回后才能在页面中使用)。
和 Vue2 相比,这种写法有一个很大的好处就是在方法中引用的时候不用再写 this 了。
ref 一般用来定义原始数据类型,像 String、Number、BigInt、Boolean、Symbol、Null、Undefined 这些。
如果你想定义对象,那么可以使用 reactive 来定义,如下:
<template>
<div>
<div>{{a}}</div>
<button @click="btnClick">clickMe</button>
<div>{{book.name}}</div>
<div>{{book.author}}</div>
</div>
</template>
<script>
import {ref, reactive} from 'vue';
export default {
name: "MyVue04",
setup() {
const a = ref(1);
const book = reactive({
name: "三国演义",
author: "罗贯中"
});
const btnClick = () => {
a.value++;
}
return {a, btnClick,book}
}
}
</script>
这里定义了 book 对象,book 对象中包含了 name 和 author 两个属性。
有的时候,你可能批量把数据定义好了,但是在访问的时候却希望直接访问,那么我们可以使用数据展开,像下面这样:
<template>
<div>
<div>{{a}}</div>
<button @click="btnClick">clickMe</button>
<div>{{name}}</div>
<div>{{author}}</div>
</div>
</template>
<script>
import {ref, reactive} from 'vue';
export default {
name: "MyVue04",
setup() {
const a = ref(1);
const book = reactive({
name: "三国演义",
author: "罗贯中"
});
const btnClick = () => {
a.value++;
}
return {a, btnClick,...book}
}
}
</script>
这样,在上面访问的时候,就可以直接访问 name 和 author 两个属性了,就不用添加 book 前缀了。
不过!!!
这种写法其实有一个小坑。
比如我再添加一个按钮,如下:
<template>
<div>
<div>{{a}}</div>
<button @click="btnClick">clickMe</button>
<div>{{name}}</div>
<div>{{author}}</div>
<button @click="updateBook">更新图书信息</button>
</div>
</template>
<script>
import {ref, reactive} from 'vue';
export default {
name: "MyVue04",
setup() {
const a = ref(1);
const book = reactive({
name: "三国演义",
author: "罗贯中"
});
const btnClick = () => {
a.value++;
}
const updateBook=()=>{
book.name = '123';
}
return {a, btnClick,...book,updateBook}
}
}
</script>
这个时候点击更新按钮,你会发现没反应!因为用了数据展开之后,响应式就失效了。所以,对于这种展开的数据,应该再用 toRefs 来处理下,如下:
<template>
<div>
<div>{{a}}</div>
<button @click="btnClick">clickMe</button>
<div>{{name}}</div>
<div>{{author}}</div>
<button @click="updateBook">更新图书信息</button>
</div>
</template>
<script>
import {ref, reactive, toRefs} from 'vue';
export default {
name: "MyVue04",
setup() {
const a = ref(1);
const book = reactive({
name: "三国演义",
author: "罗贯中"
});
const btnClick = () => {
a.value++;
}
const updateBook = () => {
book.name = '123';
}
return {a, btnClick, ...toRefs(book),updateBook}
}
}
</script>
当然,如果你将 setup 直接写在了 script 标签中,那么可以直接按照如下方式来展开数据:
<template>
<div>
<div>{{a}}</div>
<button @click="btnClick">clickMe</button>
<div>{{name}}</div>
<div>{{author}}</div>
<button @click="updateBook">更新图书信息</button>
</div>
</template>
<script setup>
import {ref, reactive, toRefs} from 'vue';
const a = ref(1);
const book = reactive({
name: "三国演义",
author: "罗贯中"
});
const btnClick = () => {
a.value++;
}
const updateBook = () => {
book.name = '123';
}
const {name, author} = toRefs(book);
</script>
5. 小结
好啦,今天就和小伙伴们分享了 Vue3 中几个新鲜的玩法~作为我们 TienChin 项目的基础(Vue 基本用法在 vhr 中都已经讲过了,所以这里就不再赘述了),当然,Vue3 和 Vue2 还有其他一些差异,这些我们都将在 TienChin 项目视频中和小伙伴们再仔细分享。
相关推荐
- 再说圆的面积-蒙特卡洛(蒙特卡洛方法求圆周率的matlab程序)
-
在微积分-圆的面积和周长(1)介绍微积分方法求解圆的面积,本文使用蒙特卡洛方法求解圆面积。...
- python创建分类器小结(pytorch分类数据集创建)
-
简介:分类是指利用数据的特性将其分成若干类型的过程。监督学习分类器就是用带标记的训练数据建立一个模型,然后对未知数据进行分类。...
- matplotlib——绘制散点图(matplotlib散点图颜色和图例)
-
绘制散点图不同条件(维度)之间的内在关联关系观察数据的离散聚合程度...
- python实现实时绘制数据(python如何绘制)
-
方法一importmatplotlib.pyplotaspltimportnumpyasnpimporttimefrommathimport*plt.ion()#...
- 简单学Python——matplotlib库3——绘制散点图
-
前面我们学习了用matplotlib绘制折线图,今天我们学习绘制散点图。其实简单的散点图与折线图的语法基本相同,只是作图函数由plot()变成了scatter()。下面就绘制一个散点图:import...
- 数据分析-相关性分析可视化(相关性分析数据处理)
-
前面介绍了相关性分析的原理、流程和常用的皮尔逊相关系数和斯皮尔曼相关系数,具体可以参考...
- 免费Python机器学习课程一:线性回归算法
-
学习线性回归的概念并从头开始在python中开发完整的线性回归算法最基本的机器学习算法必须是具有单个变量的线性回归算法。如今,可用的高级机器学习算法,库和技术如此之多,以至于线性回归似乎并不重要。但是...
- 用Python进行机器学习(2)之逻辑回归
-
前面介绍了线性回归,本次介绍的是逻辑回归。逻辑回归虽然名字里面带有“回归”两个字,但是它是一种分类算法,通常用于解决二分类问题,比如某个邮件是否是广告邮件,比如某个评价是否为正向的评价。逻辑回归也可以...
- 【Python机器学习系列】拟合和回归傻傻分不清?一文带你彻底搞懂
-
一、拟合和回归的区别拟合...
- 推荐2个十分好用的pandas数据探索分析神器
-
作者:俊欣来源:关于数据分析与可视化...
- 向量数据库:解锁大模型记忆的关键!选型指南+实战案例全解析
-
本文较长,建议点赞收藏,以免遗失。更多AI大模型应用开发学习视频及资料,尽在...
- 用Python进行机器学习(11)-主成分分析PCA
-
我们在机器学习中有时候需要处理很多个参数,但是这些参数有时候彼此之间是有着各种关系的,这个时候我们就会想:是否可以找到一种方式来降低参数的个数呢?这就是今天我们要介绍的主成分分析,英文是Princip...
- 神经网络基础深度解析:从感知机到反向传播
-
本文较长,建议点赞收藏,以免遗失。更多AI大模型应用开发学习视频及资料,尽在...
- Python实现基于机器学习的RFM模型
-
CDA数据分析师出品作者:CDALevelⅠ持证人岗位:数据分析师行业:大数据...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- idea eval reset (50)
- vue dispatch (70)
- update canceled (42)
- order by asc (53)
- spring gateway (67)
- 简单代码编程 贪吃蛇 (40)
- transforms.resize (33)
- redisson trylock (35)
- 卸载node (35)
- np.reshape (33)
- torch.arange (34)
- npm 源 (35)
- vue3 deep (35)
- win10 ssh (35)
- vue foreach (34)
- idea设置编码为utf8 (35)
- vue 数组添加元素 (34)
- std find (34)
- tablefield注解用途 (35)
- python str转json (34)
- java websocket客户端 (34)
- tensor.view (34)
- java jackson (34)
- vmware17pro最新密钥 (34)
- mysql单表最大数据量 (35)