vue2和vue3对比(语法层面)

阅读文章你将收获:

1 了解不使用组件化工具时,vue在html是如何使用的

2 知道vue2的生命周期函数有哪些

3  知道如何在组件化开发中使用vue

4  大致了解了vue2和vue3在使用上什么不同

最后:vue2和vue3除了下面我列出的有差异化的地方,还有其他地方不一样,比如vue2中只能有一个根组件,而vue3不做限制,对于一些细致的地方需要自己去详细了解........

原生的vue结构

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title>
</head>
<body><div id="app"><h3>{{ title }}</h3><div><button @click="count--">-</button><span>{{ count }}</span><button @click="count++">+</button><div>计算属性:{{doubleCount}}</div><div> 响应式数据: {{count}} </div><div><button @click="changeAge"> 修改对象的年龄</button></div></div></div><script src="https://cdn.jsdelivr.net/npm/vue@2/dist/vue.js"></script><script>const app = new Vue({// 通过el绑定对象,el里面可以填一个选择器el: '#app',// 声明响应式数据data: {count: 100,title: '计数器',obj: {name: 'jack',age: 18}},// 声明计算属性computed: {// 简单写法,属性只读doubleCount () {return this.count * 2},//  复杂写法,可以修改属性值(特殊场景使用一般不用)doubleCountPlus: {get() {return this.count * 2},set(value) {return value}}},// 声明方法methods: {saiHello () {console.log('hello');},changeAge () {// 修改对象的年龄this.obj.age = 20}},// 监视器 -监视数据变化watch: {// 简单写法count (oldValue,newValue) {console.log(oldValue,newValue);},// 完整写法 --obj: {deep: true,       //深度监视,可以监听到数据中的对象属性的变化(这里是基本变量,所以)immediate: true,  //立刻执行handler (newValue) {console.log(newValue);}}},// vue的生命周期函数// 1. 创建阶段(准备数据)beforeCreate(){console.log('数据未初始化之前'+this.count);},created(){console.log('数据完成初始化后'+this.count);},// 2. 挂载阶段(渲染模板)beforeMount(){console.log('渲染模板之前'+document.querySelector('span').innerHTML);},mounted(){console.log('渲染模板之后'+document.querySelector('span').innerHTML);},// 3. 更新阶段(修改数据 → 更新视图)beforeUpdata(){console.log('数据变化之前'+this.count);},updated(){console.log('数据变化之后'+this.count);},// 4. 卸载阶段beforeDestroyed(){console.log('对象销毁之前'+this.count);},destroyed(){console.log('对象销毁之后'+this.count);}})</script>
</body>
</html>

 vue2组件化结构

main.js

import Vue from 'vue'
import App from './App.vue'
import router from './router'
import store from './store'Vue.config.productionTip = falsenew Vue({router,store,render: h => h(App)
}).$mount('#app')

App.vue

<template><div id="app"><router-view/></div>
</template><style lang="less"></style>

vuex

import Vue from 'vue'
import Vuex from 'vuex'// 注册仓库
Vue.use(Vuex)// 导出仓库
export default new Vuex.Store({state: {},getters: {},mutations: {},actions: {},modules: {}
})

 router

import Vue from 'vue'
import VueRouter from 'vue-router'Vue.use(VueRouter)const routes = [{ path: '/home', component: () => import('@/views/home.vue') }
]const router = new VueRouter({routes
})export default router

 home组件

<template><div><div>这是home组件</div><span>666</span><div>响应式数据:{{ count }}</div><div><button @click="saiHello">调用saiHello方法</button></div><div>计算属性{{ doubleCount }}</div><div>  {{ doubleCountPlus  }}<button @click="doubleCountPlus = 60 ">修改计算属性</button></div><div><button @click="changeAge">修改obj的年龄触发监视器</button></div></div>
</template><script>
export default {name: 'homeIndex',// 定义响应式数据data () {return {count: 10,list: [],obj: {name: 'jack',age: 18}}},// 定义方法:methods: {saiHello () {console.log('hello,vue2')},changeAge () {this.obj.age++}},// 定义计算属性computed: {// 简化版doubleCount () {return this.count * 2},// 完整版  (在特定的场景使用)doubleCountPlus: {get () {return this.count * 5},set (value) {console.log(value)}}},// 定义监视器watch: {count (oldValue, newValue) {console.log(oldValue, newValue)},obj: {// 立即执行immediate: true,deep: true,handler (newValue) {console.log(newValue)}}},// vue的生命周期函数// 1. 创建阶段(准备数据)beforeCreate () {console.log('数据未初始化之前' + this.count)},created () {console.log('数据完成初始化后' + this.count)},// 2. 挂载阶段(渲染模板)beforeMount () {console.log('渲染模板之前' + document.querySelector('span').innerHTML)},mounted () {console.log('渲染模板之后' + document.querySelector('span').innerHTML)},// 3. 更新阶段(修改数据 → 更新视图)beforeUpdata () {console.log('数据变化之前' + this.count)},updated () {console.log('数据变化之后' + this.count)},// 4. 卸载阶段beforeDestroyed () {console.log('对象销毁之前' + this.count)},destroyed () {console.log('对象销毁之后' + this.count)}}
</script><style></style>

 vue3组件化结构

mian.js


import { createApp } from 'vue'
import { createPinia } from 'pinia'import App from './App.vue'
import router from './router'const app = createApp(App)app.use(createPinia())
app.use(router)app.mount('#app')

App.vue

<script setup>
</script><template><RouterView />
</template><style scoped></style>

router(路由)

import { createRouter, createWebHistory } from 'vue-router'const router = createRouter({history: createWebHistory(import.meta.env.BASE_URL),routes: [{path: '/home', component: () => import('@/views/home.vue')},{path: '/login', component: () => import('@/views/login.vue')}]
})export default router

pinia(仓库)

import { ref, computed } from 'vue'
import { defineStore } from 'pinia'export const useCounterStore = defineStore('counter', () => {const count = ref(10)const doubleCount = computed(() => count.value * 2)function increment() {count.value++}return { count, doubleCount, increment }
})

 vue3_home组件

<script setup>
import { computed, onBeforeMount, onBeforeUnmount, onMounted, onUnmounted, onUpdated, reactive, ref, watch } from "vue";
import testComponent from '@/components/testComponent.vue'// 导入仓库内容
import { useCounterStore } from "@/stores/counter";
import { onBeforeRouteLeave, useRoute, useRouter } from "vue-router";
// 实例化仓库对象
const counter = useCounterStore()
console.log('输出仓库对象count' + counter.count)const route = useRoute()
const router = useRouter()
const login = () => {router.push(`/login?name=${obj.name}`)
}
//  定义响应式数据
const count = ref(10)
// 使用reactive定义的响应式数据可以直接通过对象获取属性obj.name
const obj = reactive({age: 18,name: '李四'
})// 定义方法
const saiHello = () => {console.log('hello,vue3');
}
function changeCount () {count.value++
}
const changeName = () => {obj.age++
}//定义计算属性
const doubleCount = computed(() => {return count.value * 3
})
// 计算属性的完整写法
const doubleCountPlus = computed({get () {return count.value * 4},set (value) {return count.value * value}
})//watch 监视器 watch(obj,(oldValue,newValue) => {console.log(oldValue, newValue);
},{deep: true,immediate: true
})// vue的生命周期函数// 这二个方法在vue3中被setup语法糖替代了// function beforeCreate(){// console.log('数据未初始化之前'+this.count);// }// function created(){// console.log('数据完成初始化后'+this.count);// }// 2. 挂载阶段(渲染模板)onBeforeMount(() => {console.log('渲染模板之前');}) onMounted(() => {console.log('渲染模板之后');})// 3. 更新阶段(修改数据 → 更新视图)onBeforeRouteLeave(() => {console.log('数据变化之前');})onUpdated(() => {console.log('数据变化之后');})// 4. 卸载阶段onBeforeUnmount(() => {console.log('对象销毁之前');})onUnmounted(() => {console.log('对象销毁之前');})
</script><template><div>我是home页面</div><div><button @click="saiHello">saiHello</button></div><div>count变量:{{ count }} <button @click="changeCount">+</button> </div><div>计算属性:{{ doubleCount }} </div><div>age:<button @click="changeName">+</button></div><div>obj:{{obj}}</div><testComponent>v-我是默认插槽的内容<template v-slot:test>我是具名插槽的内容</template></testComponent><div>{{ counter.count }}</div><button @click="login">点击按钮跳转到登录页</button></template><style scoped></style>

 

vue2和vue3不同点总结

指令

  • 相同点: v-html,v-show,v-if ,v-else,v-esle-if,v-bind,v-on 

        这七个指令的用法和简写方式不管是在vue2还是vue3用法都是一样的

  • 不同点:

        v-model

                作用于表单元素,用法一样

        作用于其他元素

                vue2       ===                      :value属性input事件

                vue3:   =====                  :valuemode属性和:update:modelValue事件

响应式数据

在vue中,要想将变量定义成响应式数据,需要将变量定义到data中

在vue中如果要定义响应式数据,需要使用ref()和reactive()这二个函数来定义

methods方法

在vue2中,所有的不同方法都需要写在选项methos中

在vue3引入了组合式api的概念,只要在<script setup></script>加入了setup这个语法糖,那么就可以将普通方法和变量定义到script的里面

computed计算属性

在vue2中,要定义计算属性,需要将方法或者对象定义到computed选项中

在vue3中,要定义计算属性,需要将方法或者对象定义到computed()函数中

watch监听器

在vue2中,要定义计算属性,需要将方法或者对象定义到watch选项中

在vue3中,要定义计算属性,需要将方法或者对象定义到watch()函数中

生命周期函数

组件通信

父子关系的组件通信:

相同点:同样是使用props和emit来实现

不同点:在vue中要获取pros和emit这二个对象需要通过defineProps和defineEmits这二个自定义宏来获取

获取dom元素

在vue2中使用ref和refs来绑定和获取dom元素

在vue3获取绑定对象

声明变量

 绑定dom元素

插槽

vue2和vue3中插槽的用法是一样的

声明:

使用:

路由

相同点:

1  都是使用vue-router这个插件来实现路由来切换页面

2   声明式导航 (router-link)的用法是一样的

不同点:

  • 初始化路由的方式不一样

  • 编程路由获取和使用方式不一样

 vue2:通过$route和$router(都是内置的对象)

获取路由参数: $route

编程式路由跳转:$router

vue3:通过useRoute和useSouter这二个函数来获取route和router对象

仓库:

vuex

vue2中使用vuex来管理 vue 通用的数据 (多组件共享的数据)

官网:Vuex官网

定义:

使用:

获取仓库变量:

调用仓库的mutatios方法 

 

调用仓库的actions方法

带哦用仓库的getters属性

pinia

vue3中使用pinia来管理 vue 通用的数据 (多组件共享的数据)

官方文档: Pinia

pinia和vuex的用法用法基本一样,但不同的是在pinia中没有moudel和mutatios这二个选项

,然后使用的方式也有一些小差别,想了解更多可以查看上面的官网

声明:

使用:

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

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

相关文章

高防服务器的价格受到哪些因素的影响?

高防服务器可以帮助网站拒绝服务攻击&#xff0c;能够进行定时扫描网络节点&#xff0c;并且查找可能会存在的安全漏洞的服务器类型&#xff0c;对于经常遭受到DDOS攻击的大型游戏网络企业会选择高防服务器。 那面对价格多样的高防服务器我们应该怎样进行选择&#xff0c;高防服…

day41打卡

day41打卡 46. 携带研究材料&#xff08;第六期模拟笔试&#xff09; 状态表示 ​ 二维&#xff1a;dp[i] [j] 表示从下标为[0-i]的物品里任意取&#xff0c;放进容量为j的背包&#xff0c;价值总和最大是多少。 一维&#xff1a; ​ dp[j]表示&#xff1a;容量为j的背包&a…

模型 HBG(品牌增长)

系列文章 分享 模型&#xff0c;了解更多&#x1f449; 模型_总纲目录。品牌增长法。 1 HBG(品牌增长)模型的应用 1.1 江小白使用HBG模型提高品牌知名度和销售额 选择受众市场&#xff1a;江小白的目标客户是年轻人&#xff0c;他们喜欢简单、时尚的产品。因此&#xff0c;江…

面试前端性能优化八股文十问十答第三期

面试前端性能优化八股文十问十答第三期 作者&#xff1a;程序员小白条&#xff0c;个人博客 相信看了本文后&#xff0c;对你的面试是有一定帮助的&#xff01;关注专栏后就能收到持续更新&#xff01; ⭐点赞⭐收藏⭐不迷路&#xff01;⭐ 1&#xff09;如何⽤webpack来优化…

数据结构D4作业

1.实现单向循环链表的功能 loop.c #include "loop.h" loop_p create_loop() { loop_p H(loop_p)malloc(sizeof(loop)); if(HNULL) { printf("创建失败\n"); return NULL; } H->len0; H->nextH; ret…

一文彻底搞懂SQL优化

文章目录 1. 索引优化2. WHERE 子句优化3. JOIN 优化4. 查询结果优化5. 子查询优化6. 数据库统计信息和缓存优化7. 事务优化8. 数据库配置优化9. 使用适当的数据类型10. 监控和调优 SQL(Structured Query Language&#xff09;优化是指对 SQL 查询语句进行调整和改进&#xff0…

关于 Reflect 的笔记

背景&#xff1a;Reflect 为了操作对象而提供的新Api 和 Proxy对象一样 特点 将object 对象的一些明显属于语言内部的方法&#xff0c;放到Reflect 上处理&#xff1b;修改某些object返回的异常结果&#xff0c;让其变得更合理&#xff1b;让object操作都变成函数行为&#xf…

基于ElementUI封装省市区四级联动下拉选择

基于ElementUI封装的省市区下拉级联选择 效果 数据 最新省市区JSON数据获取&#xff1a;https://xiangyuecn.github.io/AreaCity-JsSpider-StatsGov/ 参数说明 参数说明inputNumShow下拉框的数量&#xff0c;最多4个defaultAddress默认显示省市区 例&#xff1a;[‘安徽’, …

按形如 a*sqrt(b) 的格式输出一个非负整数的平方根

【题目描述】 输入一个非负整数 x&#xff0c;若能完全开平方根&#xff0c;则输出其对应的整数平方根值。 否则&#xff0c;按形如 a*sqrt(b) 的格式输出其平方根值&#xff08;a 与 b 均为整数&#xff0c;且 a≠1&#xff0c;b≠1&#xff09;。【输入输出】 典型的输入输出…

【C++初阶】--类和对象(下)

目录 一.const成员 1.权限放大问题 2.权限的缩小 二.再谈构造函数 1.构造函数体赋值 2.初始化列表 (1)概念 (2)使用 ①在对象实例化过程中&#xff0c;成员变量先依次进行初始化 ②再进行函数体内二次赋值 3.explicit关键字 (1)C为什么要存在自动隐式类型转换…

Linux内核中自旋锁驱动代码举例二

一. 简介 前面学习了不考虑中断的自旋锁的代码举例,文章地址: Linux内核自旋锁驱动代码举例一-CSDN博客 但是在 Linux系统中,中断时存在的。所以,这里学习使用带保存中断状态的自旋锁API函数,实现对Led设备的互斥访问。 二. 带保存中断状态的自旋锁函数使用 1. 准备…

Linux第64步_编译移植好的虚拟机文件

最好还是认真了解linux系统移植的整个过程&#xff0c;否则&#xff0c;可能会让你误入歧途。 1、编译移植好的tf-a 1)、编译生成“tf-a-stm32mp157d-atk-trusted.stm32” 输入“cd /home/zgq/linux/atk-mp1/tfa/my-tfa/tf-a-stm32mp-2.2.r1/回车”&#xff0c;切换到“/hom…

算法打卡day1|数组篇|Leetcode 704.二分查找、27.移除元素

数组理论基础 数组是存放在连续内存空间上的相同类型数据的集合&#xff0c;可以方便的通过下标索引的方式获取到下标下对应的数据。 1.数组下标都是从0开始的。 2.数组内存空间的地址是连续的。 正是因为数组的在内存空间的地址是连续的&#xff0c;所以我们在删除或者增添…

算法题--华为od机试考试(分苹果、字符串统计及重排、高矮个子排队)

目录 分苹果 题目描述 输入描述 输出描述 示例1 输入 输出 备注 示例2 输入 输出 解析 答案 字符统计及重排 题目描述 输入描述 输出描述 示例1 输入 输出 说明 示例2 输入 输出 说明 解析 答案 高矮个子排队 题目描述 输入描述 输出描述 备注…

【深度学习笔记】3_5 图像分类数据集fashion-mnist

注&#xff1a;本文为《动手学深度学习》开源内容&#xff0c;仅为个人学习记录&#xff0c;无抄袭搬运意图 3.5 图像分类数据集&#xff08;Fashion-MNIST&#xff09; 在介绍softmax回归的实现前我们先引入一个多类图像分类数据集。它将在后面的章节中被多次使用&#xff0c…

STM32使用PB3, PB4引脚的注意事项

STM32的PB3, PB4引脚作为GPIO引脚需要注意&#xff0c;因为他们默认分别是JTDO和NJTRST引脚。 笔者在设计可调增益增益放大器&#xff08;VGA&#xff09;的时候&#xff0c;使用4个GPIO读取外部控制电压&#xff0c;根据约定的编码格式设定DAC的输出电压&#xff0c;从而设置V…

《Docker 简易速速上手小册》第1章 Docker 基础入门(2024 最新版)

文章目录 1.1 Docker 简介与历史1.1.1 Docker 基础知识1.1.2 重点案例&#xff1a;Python Web 应用的 Docker 化1.1.3 拓展案例 1&#xff1a;使用 Docker 进行 Python 数据分析1.1.4 拓展案例 2&#xff1a;Docker 中的 Python 机器学习环境 1.2 安装与配置 Docker1.2.1 重点基…

消息队列-RabbitMQ:发布确认—发布确认逻辑和发布确认的策略

九、发布确认 1、发布确认逻辑 生产者将信道设置成 confirm 模式&#xff0c;一旦信道进入 confirm 模式&#xff0c;所有在该信道上面发布的消息都将会被指派一个唯一的 ID (从 1 开始)&#xff0c;一旦消息被投递到所有匹配的队列之后&#xff0c;broker 就会发送一个确认给…

Python基础教程——17个工作必备的Python自动化代码

您是否厌倦了在日常工作中做那些重复性的任务&#xff1f;简单但多功能的Python脚本可以解决您的问题。 引言 Python是一种流行的编程语言&#xff0c;以其简单性和可读性而闻名。因其能够提供大量的库和模块&#xff0c;它成为了自动化各种任务的绝佳选择。让我们进入自动化…

K8s环境搭建

一、基础环境准备 VMware虚拟机&#xff0c;安装三台CentOS&#xff0c;网络环境选择NAT模式&#xff0c;推荐配置如下&#xff08;具体安装步骤省略&#xff0c;网上很多虚拟机安装CentOS7的教程&#xff09; 二、网络环境说明 使用NAT模式&#xff0c;我的IP分别是&#xf…