组件可以拥有多个根节点,解决多个div嵌套问题
Vue3的数据获取在反应数据(Reactive Data)中,通常包含在一个反应状态(Reactive State)变量中。(reactive的作用是将对象包装成响应式对象,通过Proxy代理后的对象。)
响应式原理: Proxy代理拦截:reactive函数执行,会将传入的target对象通过Proxy包装,拦截它的get,set等,并将代理的target缓存。(组件在render阶段,视图会读取数据对象上的值进行渲染,此时便触发了Proxy的get,由此触发对应的track函数,记录下了对应的ReactiveEffect,也就是常说的依赖收集。)
1).Ref与reactive区别:
reactive 和 ref 都是用来定义响应式数据的, reactive更推荐去定义复杂的数据类型, ref 更推荐定义基本类型,ref 和 reactive 本质我们可以简单的理解为ref是对reactive的二次包装, ref定义的数据访问的时候要多一个.value。(ref定义基本数据类型,ref也可以定义数组和对象)
2).判断数据到底是ref还是reactive?
通过isRef / isReactive 方法,对当前数据的__v_ref来判断的,如果有这个私有的属性, 并且取值为true, 那么就代表是一个ref类型的数据
在Vue3.0,我们就需要使用一个新的setup()方法,此方法在组件初始化构造的时候触发。
使用以下三步来建立反应性数据:
1)从vue引入reactive
2)使用reactive()方法来声名我们的数据为反应性数据
3)使用setup()方法来返回我们的反应性数据,从而template可以获取这些反应性数据
import { reactive } from 'vue'
export default {
props: {
title: String
},
setup () {
const state = reactive({
username: '',
password: ''
})
return { state }
}}
这里构造的反应性数据就可以被template使用,可以通过state.username和state.password获得数据的值,需要先声名一个方法然后在setup()方法中返回(return), 在组件内就可以调用这个方法了。
Vue3 的合成型API里面的setup()方法也是可以用来操控methods的。创建声名方法其实和声名数据状态是一样的。需要先声名一个方法然后在setup()方法中返回(return), 这样组件内就可以调用这个方法了。
export default {
props: {
title: String
},
setup () {
const state = reactive({
username: '',
password: ''
})
const login = () => {
// 登陆方法 }
return {
login,
state
}
}}
现在 Vue3 的合成型API里面的setup()方法可以包含了基本所有东西。生命周期的钩子就是其中之一!但是在 Vue3 生周期钩子不是全局可调用的了,需要另外从vue中引入,生命周期的挂载钩子叫onMounted,引入后我们就可以在setup()方法里面使用onMounted挂载的钩子了。(与2.0生命周期相似,在前面加了on,setup这个生命周期发生在beforeCreate和created之前,此时无法访问data、computed、methods、refs )
Vue3 的设计模式给予开发者们按需引入需要使用的依赖包,在 Vue3 使用计算属性,需要在组件内引入computed。使用方式就和反应性数据(reactive data)一样,在state中加入一个计算属性。(计算属性可以写在声明的响应式变量内,也可单独声明使用)
setup () {
const state = reactive({
username: '',
password: '',
lowerCaseUsername: computed(() => state.username.toLowerCase())
})
const twiceTheCounter = computed(() => counter.value * 2)
接收组件props参数传递这一块为我们带来了Vue2和Vue3之间最大的区别。在 Vue2,this代表的是当前组件,不是某一个特定的属性。所以我们可以直接使用this访问prop属性值。就比如下面的例子在挂载完成后打印处当前传入组件的参数title。在 Vue3 中,this无法直接拿到props属性,emit events(触发事件)和组件内的其他属性。不过全新的setup()方法可以接收两个参数: 1.props - 不可变的组件参数
2.context - Vue3 暴露出来的属性(emit,slots,attrs)
setup (props,context) {
//Context的使用(// attrs (非响应式对象,等同于 $attrs用于简介传参,slots 插槽 非响应式对象,等同于 $slots,emit触发事件方法,等同于 $emit,expose 暴露公共property 函数,将setup里的内容暴露到外部)
onMounted(() => {
console.log('title: ' + props.title)
})
一个项目使用
//父组件里的子组件
//子组件的内容
{{title}}
为了获得对模板内元素或组件实例的引用,一样声明 ref 并从 setup() 返回:
This is a root element
这里在渲染上下文中暴露root,并通过ref=”root”,将其绑定到div作为其ref。(在虚拟dom补丁算法中,若Vnode的ref键对应于渲染上下文中的 ref,则 VNode 的相应元素或组件实例将被分配给该 ref 的值。这是在虚拟 DOM 挂载/打补丁过程中执行的,因此模板引用只会在初始渲染之后获得赋值,即在onMounted生命周期中)
9.watch监听
使用从 Vue 导入的 watch 函数执行相同的操作。它接受 3 个参数:
import { watch } from 'vue'
//一个变量
const counter = ref(0)
//监听函数 watch(监听的值,回调,可选的配置选项)
watch(counter, (newValue, oldValue) => {
console.log('The new counter value is: ' + counter.value)
})
注意:侦听模板引用的变更可以替代前面例子中演示使用的生命周期钩子。
但与生命周期钩子的一个关键区别是watch() 和 watchEffect()在 DOM 挂载或更新之前运行副作用,所以当侦听器运行时,模板引用还未被更新。因此,使用模板引用的侦听器应该用 flush: 'post' 选项来定义,这将在 DOM 更新后运行,确保模板引用与 DOM 保持同步。
watchEffect(() => {
console.log(root.value) // => This is a root element
},
{
flush: 'post'
})
简述:Mixin 提供了一种非常灵活的方式,来分发 Vue 组件中的可复用功能。一个 mixin 对象可以包含任意组件选项。当组件使用 mixin 对象时,所有 mixin 对象的选项将被“混合”进入该组件本身的选项。
局部mixins的配置:
export const mixin1 = {
methods: {
showName(){
alert('第一个mixin')
}
},
mounted() {
console.log('你好啊!')
},}
export const mixin2 = {
data() {
return {
sum:100
}
},
methods: {
showName(){
alert('第二个mixin')
}
},
}
全局注册:全局mixins 对vm,和所有vm均进行mixins
//引入Vue
import Vue from 'vue'
//引入App
import App from './App.vue'
import {mixin1,mixin2} from '../mixin'
//关闭Vue的生产提示
Vue.config.productionTip = false
Vue.mixin(mixin1)
Vue.mixin(mixin2)
//创建vm
new Vue({
el:'#app',
render: h => h(App)
})
除了核心功能默认内置的指令 (例如 v-model 和 v-show),Vue 也允许注册自定义指令。注意,在 Vue 中,代码复用和抽象的主要形式是组件。然而,有的情况下,你仍然需要对普通 DOM 元素进行底层操作,这时候就会用到自定义指令。
举个聚焦输入框的例子,如下:
//全局注册
const app = Vue.createApp({})// 注册一个全局自定义指令 `v-focus`
app.directive('focus', {
// 当被绑定的元素挂载到 DOM 中时……
mounted(el) {
// 聚焦元素
el.focus()
}})
注册局部指令:
directives: {
focus: {
// 指令的定义
mounted(el) {
el.focus()
}
}}
注册局部指令:
directives: {
focus: {
// 指令的定义
mounted(el) {
el.focus()
}
}}
使用方式:
使用方式
传参和原来一样,取参发生改变:
Params传参:
点击跳转到home页面并传参
vue3.0中import { useRoute } from 'vue-router'
const route = useRoute()
console.log(route.params.id);//111
query传参:
test
//接受的时候
import { useRoute } from 'vue-router'
const route = useRoute()
console.log(route.query.id);//999
Js传参方式:
import { useRouter } from 'vue-router'
const route = useRouter()
route.push({
path:"/lianxi",
query:{
id:666
}
});
route.push({
name:'lianxi',
params:{
id:666
}
});
import { useRoute } from 'vue-router'
const route = useRoute()
console.log(route.query.id);//999
现在基本都看到vue2与vue3其实概念与理念都是一样的。只是有一些属性获取方式和声名和定义方式稍微变了。总结一下,我觉得 Vue3 给我们前端开发者带来了全新的开发体验,更好的使用弹性,可控度也得到了大大的提升。全新的合成式API(Composition API)可以提升代码的解耦程度 —— 特别是大型的前端应用,效果会更加明显。还有就是按需引用的有了更细微的可控性,让项目的性能和打包大小有更好的控制。
案例项目代码整理:
{{ state.title }}
Values: {{ state.username + ' ' + state.password }}