vue3.0 上手体验


Posted in Javascript onSeptember 21, 2020

vue3.0 beta 版本已经发布有一阵子了,是时候上手体验一波了~

注意,本文所有演示都是基于 vue3.0 beta 版本,不保证后续正式版 api 不改动。等官方文档出来后,以官网为准。

环境搭建

直接使用脚手架,如果本地没有安装的可以执行脚手架安装命令:

npm install -g @vue/cli

如果本地安装过的,可以尝试更新一下:

npm update -g @vue/cli

测试 vue-cli 版本:

vue -V
@vue/cli 4.4.1

接下来创建一个vue项目:

vue create vue3-demo

在出现的命令交互窗口选择 Manually select features :

Vue CLI v4.4.1
? Please pick a preset:
 常用配置 (router, vuex, sass, babel, eslint)
 sass (router, vuex, sass, babel, eslint)
 test (less, babel, eslint)
 default (babel, eslint)
❯ Manually select features

随后勾选以下选项,一般开发商业项目都需要这些:

Vue CLI v4.4.1
? Please pick a preset: Manually select features
? Check the features needed for your project:
 ◉ Babel
 ◯ TypeScript
 ◯ Progressive Web App (PWA) Support
 ◉ Router
 ◉ Vuex
❯◉ CSS Pre-processors
 ◉ Linter / Formatter
 ◯ Unit Testing
 ◯ E2E Testing

回车后根据自己的习惯选择好,就开始创建项目。注意这时候还是 vue2 的项目环境,接下来就是升级为 vue3 的运行环境。

升级为 vue3.0 项目

vue-cli 还没有直接支持 vue3.0,需要依赖插件升级,输入指令:

cd vue3-demo
vue add vue-next

执行完上述命令后,会自动安装 vue-cli-plugin-vue-next 插件,它会将项目升级为 vue3.0 的依赖环境,包括 vue-router 和 vuex 都会升级为 4.x 的版本。

最新版 vue-cli 可以直接创建 vue3 项目了(2020.09更新)

今天将 vue-cli 更新到了 4.5.4 版本,发现可以直接创建 vue3 项目了。使用 vue create vue3-demo 命令创建一个测试项目,会出现以下选项:

vue3.0 上手体验

 可以直接选择默认的 vue3 配置,也可以选择最后一项来手动配置,这里演示手动配置,选择 Manually select features,回车:

vue3.0 上手体验

根据需要选择,注意第一项表明了可以选择 vue 版本,选完后回车:

vue3.0 上手体验

这里选择 vue 的版本,直接选择 3.x 回车即可。这样就能创建一个基于 vue3 搭建的项目了。

vue3.0 特性体验

按照上面步骤升级为 vue3.0 项目后,会自动帮我们将一些文件改成 vue3.0 的写法。

创建vue实例

vue3 创建vue实例不需要使用 new 的方式了,来看 src/main.js 文件:

import { createApp } from 'vue'
import App from './App.vue'
import router from './router'
import store from './store'

createApp(App).use(router).use(store).mount('#app')

现在是函数式风格来创建vue实例,还记得 vue2 是怎么创建实例的吗,对比下:

// Vue2 创建实例

import Vue from 'vue'
import App from './App'
import router from './router'
import store from './store'

new Vue({
 el: '#app',
 router,
 store,
 render: h => h(App)
})

你喜欢哪一种方式?我比较喜欢vue3.0的函数式风格。

路由

看看路由配置文件:src/router/index.js

import { createRouter, createWebHistory } from 'vue-router'
import Home from '../views/Home.vue'

const routes = [
 {
 path: '/',
 name: 'Home',
 component: Home
 },
 {
 path: '/about',
 name: 'About',
 // route level code-splitting
 // this generates a separate chunk (about.[hash].js) for this route
 // which is lazy-loaded when the route is visited.
 component: () => import(/* webpackChunkName: "about" */ '../views/About.vue')
 }
]

const router = createRouter({
 history: createWebHistory(process.env.BASE_URL),
 routes
})

export default router

这是升级后路由配置方法,可以看到与 vue2 版本有很大的区别。现在创建路由实例需要手动引入 createRouter 方法,创建 history 模式路由也需要手动引入 createWebHistory 方法,这达到 Tree-Shaking 的目的,即不会把所有的 api 都打包进来,只会打包你用到的 api,vue3 将都会使用这种形式。 

响应式数据、methods、watch 和 computed

这里应该是改动最大的部分,也是争议最大的部分,来看看怎么回事。

在vue2版本中,我们声明响应式数据是这样的:

// Vue2
export default {
 // ....
 data() {
 return {
 state: {
 count: 0
 }
 };
 },
}

在vue3.0中,需要这样:

// Vue3
import { ref } from 'vue'
export default {
 setup () {
 const count = ref(0) // 声明 count,初始值为 0
 const str = ref('hello') // 声明 str,初始值为 'hello'
 return {
 count,
 str
 }
 }
}

我们要先引入 ref 方法,然后使用它来声明响应式变量。重要的是,这些操作需要在 setup 函数中进行,而且添加 methods,watch 和 computed 都需要在 setup 中进行。这就跟在vue2中有很大的不同,vue2中我们是使用选项的方式来创建 data、methods、watch 和 computed 的。 

接下来演示一个计数器的功能,结合 methods、watch 和 computed:

<template>
 <div class="home">
 <p>count: {{count}}</p>
 <p>doubleCount: {{doubleCount}}</p>
 <button @click="add">增加</button>
 </div>
</template>

<script>
import { ref, watch, computed } from 'vue'
export default {
 setup () {
 // 声明 count 变量,初始值为 0
 const count = ref(0)

 // 定义 add 方法
 const add = () => {
 count.value++
 }

 // 定义 watch,需要手动引入 watch 方法
 watch(
 () => count.value,
 (val, oldVal) => { console.log(`new count: ${val},old count: ${oldVal}`) }
 )

 // 定义computed,同样需要手动引入 computed 方法
 const doubleCount = computed(() => count.value * 2)

 return {
 count,
 add,
 doubleCount
 }
 }
}
</script>

来看这个例子,首先定义方法是可以直接定义在 setup 函数中的,然后 return 出去即可,不知可否注意到在方法里访问 count 时,是使用 .value 方法访问的,这里要强调一下,在 setup 中访问已声明的响应式变量时,需要使用 .value 方式访问,包括在 watch 和 computed 中。

接下来是定义 watch,需要手动引入 watch 方法,这也是达到了 Tree-Shaking 的目的,使用到什么就引入什么,最后打包也只打包用到的 api,后面的 computed 也同理。

watch方法有两个参数,都是函数,第一个函数返回要监听的值,第二个函数是回调函数,它两个参数分别表示新值和旧值。

computed 方法返回计算过的值,最后需要 return 出去。用法如上,还是比较好理解的。

你也可以这样声明响应式数据(使用 reactive)

前面说到声明响应式数据,需要使用 ref,其实你也可以使用 reactive 来一次声明多个变量,下面例子:

<template>
 <div class="home">
 <p>str: {{state.str}}</p>
 <p>count: {{state.count}}</p>
 <button @click="add">增加</button>
 </div>
</template>

<script>
import { reactive } from 'vue'
export default {
 setup () {
 // 引入 reactive,同时定义多个变量
 const state = reactive({
 count: 0,
 str: 'hello'
 })

 // 现在访问变量,不能使用 .value 方式访问了
 const add = () => {
 // state.count.value++ // 错误
 state.count++
 }

 return {
 state,
 add
 }
 }
}
</script>

reactive 和 ref

上面说到 ref 和 reactive,这里再简单说说。reactive 是接收一个普通对象,返回该对象的响应式代理,它等同于 2.x 中的 Vue.observable()。

const obj = reactive({ count: 0 })

// obj 此时是一个响应式的对象
// 访问或修改,直接基于 obj.count

ref 也是接收一个参数并返回一个响应式且可改变的 ref 对象,一般参数是基础类型,比如数值或字符串等。如果传入的参数是一个对象,将会调用 reactive 方法进行深层响应转换。ref 对象拥有一个指向内部值的单一属性 .value,即当你要访问它的值时,需要 .value 拿到它的值。但是如果是在 setup 中返回且用到模板中时,在 {{}} 里不需要加 .value 访问,在返回时已经自动解套。

<template>
 <div>{{ count }}</div>
</template>

<script>
 export default {
 setup() {
 return {
 count: ref(0), // 这里返回,在模板中无需 .value 访问值
 }
 },
 }
</script>

获取路由信息

vue3.0 中使用 getCurrentInstance 方法获取当前组件实例,然后通过 ctx 属性获取当前上下文,ctx.$router 是路由实例,而 ctx.$router.currentRoute 就包含当前路由信息。

<script>
import { getCurrentInstance } from 'vue'
export default {
 setup () {
 const { ctx } = getCurrentInstance()
 console.log(ctx.$router.currentRoute.value)
 }
}
</script>

vuex

查看文件 src/store/index.js:

import Vuex from 'vuex'

export default Vuex.createStore({
 state: {
 },
 mutations: {
 },
 actions: {
 },
 modules: {
 }
})

发现创建 store 实例的方式改变了,vue2 中是使用 new 的方式:

// Vue2 中创建 store 实例
export default new Vuex.Store({
 // ... 
})

一个小例子演示 vue3.0 中使用 store。

创建 store:

import Vuex from 'vuex'

export default Vuex.createStore({
 state: {
 count: 0
 },
 mutations: {
 ADD (state) {
 state.count++
 }
 },
 actions: {
 add ({ commit }) {
 commit('ADD')
 }
 },
 modules: {
 }
})

组件中使用 store:

<template>
 <div class="home">
 <p>{{count}}</p>
 <button @click="add">增加</button>
 </div>
</template>

<script>
import { computed } from 'vue'
import { useStore } from 'vuex'
export default {
 setup () {
 const store = useStore()
 const count = computed(() => store.state.count)

 const add = () => {
 store.dispatch('add')
 }

 return {
 count,
 add
 }
 }
}
</script>

可以看到 vuex 的 api 基本没变化,只是在组件中使用时需要引入 useStore 方法返回 store 实例,其实你也可以在当前组件上下文中获取 store,如下:

import {getCurrentInstance} from 'vue'

// ...
const store = getCurrentInstance().ctx.$store

 大概就记录到这吧,基本涵盖到了日常使用的方面。等待 vue3.0 的正式版吧。(还是那句话,上面所讲只是基于目前 vue3.0 beta 版本,不保证后续 api 不改动,等正式版官方文档吧)

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
JS类中定义原型方法的两种实现的区别
Mar 08 Javascript
JS 对象介绍
Jan 20 Javascript
google jQuery 引用文件,jQuery 引用地址集合(jquery 1.2.6至jquery1.5.2)
Apr 24 Javascript
jquery实现带复选框的表格行选中删除时高亮显示
Aug 01 Javascript
利用CSS、JavaScript及Ajax实现高效的图片预加载
Oct 16 Javascript
在百度知道团队中快速审批新成员的js脚本
Feb 02 Javascript
JavaScript jquery及AJAX小结
Jan 24 Javascript
js删除Array数组中指定元素的两种方法
Aug 03 Javascript
JavaScript数据类型学习笔记分享
Sep 01 Javascript
ES6新特性六:promise对象实例详解
Apr 21 Javascript
Angular自定义组件实现数据双向数据绑定的实例
Dec 11 Javascript
vue.js $refs和$emit 父子组件交互的方法
Dec 20 Javascript
微信小程序实现点击页面出现文字
Sep 21 #Javascript
vue pages 多入口项目 + chainWebpack 全局引用缩写说明
Sep 21 #Javascript
理解JavaScript中的Proxy 与 Reflection API
Sep 21 #Javascript
vue实践---根据不同环境,自动转换请求的url地址操作
Sep 21 #Javascript
vue实践---vue不依赖外部资源实现简单多语操作
Sep 21 #Javascript
js节流防抖应用场景,以及在vue中节流防抖的具体实现操作
Sep 21 #Javascript
微信小程序实现翻牌抽奖动画
Sep 21 #Javascript
You might like
php 中文字符入库或显示乱码问题的解决方法
2010/04/12 PHP
php实现读取手机客户端浏览器的类
2015/01/09 PHP
微信公众号开发之文本消息自动回复php代码
2016/08/08 PHP
Laravel 6.2 中添加了可调用容器对象的方法
2019/10/22 PHP
Nginx+php配置文件及原理解析
2020/12/09 PHP
javascript 用原型继承来实现对象系统
2010/03/22 Javascript
通过正则格式化url查询字符串实现代码
2012/12/28 Javascript
Jquery submit()无法提交问题
2013/04/21 Javascript
JavaScript基于ajax编辑信息用法实例
2015/07/15 Javascript
手机端转盘抽奖代码分享
2015/09/10 Javascript
详解javascript函数的参数
2015/11/10 Javascript
JS仿hao123导航页面图片轮播效果
2016/09/01 Javascript
Jquery on绑定的事件 触发多次实例代码
2016/12/08 Javascript
通过BootStrap-select插件 js jQuery控制select属性变化
2017/01/03 Javascript
[51:22]Fnatic vs IG 2018国际邀请赛小组赛BO2 第一场 8.17
2018/08/18 DOTA
Python实现的十进制小数与二进制小数相互转换功能
2017/10/12 Python
Python装饰器用法实例总结
2018/05/26 Python
python中的turtle库函数简单使用教程
2018/07/23 Python
python组合无重复三位数的实例
2018/11/13 Python
Python多线程实现支付模拟请求过程解析
2020/04/21 Python
Python3爬虫中Selenium的用法详解
2020/07/10 Python
python中用Scrapy实现定时爬虫的实例讲解
2021/01/18 Python
台湾网购生鲜第一品牌:i3Fresh爱上新鲜
2017/10/26 全球购物
地球上最先进的胡子和头发修剪器:Bevel
2018/01/23 全球购物
英国广泛的照明产品网站:Lights4living
2018/01/28 全球购物
超市实习总结自我鉴定
2013/09/19 职场文书
最新大学职业规划书范文
2013/12/30 职场文书
亲属关系公证书
2014/04/08 职场文书
8和9的加减法教学反思
2014/05/01 职场文书
文明和谐家庭事迹材料
2014/05/18 职场文书
2014年禁毒工作总结
2014/11/24 职场文书
2014年妇产科工作总结
2014/12/08 职场文书
学校教学管理制度
2015/08/06 职场文书
低端且暴利的线上线下创业项目分享
2019/09/03 职场文书
原生JS封装vue Tab切换效果
2021/04/28 Vue.js
Win11运行育碧游戏总是崩溃怎么办 win11玩育碧游戏出现性能崩溃的解决办法
2022/04/06 数码科技