目录

前言

setup语法糖

一、基本用法

二、data和methods

三、计算属性computed

四、监听器watch、watchEffect

五、自定义指令directive

六、import导入的内容可直接使用

七、声明props和emits?

八、父组件获取子组件的数据

?九、provide和inject传值

十、路由useRoute和useRouter

十一、对await异步的支持

十二、nextTick

十三、全局属性globalProperties

十四、生命周期

十五、与普通的script标签一起使用

十六、v-memo新指令

style新特性

一、局部样式

二、深度选择器

三、插槽选择器

四、全局选择器

五、混合使用局部与全局样式

六、支持CSS?Modules

七、与setup一同使用

八、动态 CSS


前言

满满的干货,建议收藏慢慢看,可以当作Vue3.0的学习资料。

在vue2.0时期,组件里定义的各类变量、方法、计算属性等是分别存放到data、methods、computed…选项里,这样编写的代码不便于后期的查阅(查找一个业务逻辑需要在各个选项来回切换)。setup函数的推出就是为了解决这个问题,让新手开发者更容易上手…

setup语法糖

setup是Vue3.0后推出的语法糖,并且在Vue3.2版本进行了大更新,像写普通JS一样写vue组件,对于开发者更加友好了;按需引入computed、watch、directive等选项,一个业务逻辑可以集中编写在一起,让代码更加简洁便于浏览。

一、基本用法

只需在里的代码编译成一个setup函数

<script setup>
console.log('hello script setup')
</script>

普通的

二、data和methods

<template><h1>{{ msg }}</h1><p>{{obj.key}}</p><input v-model="num" type="text" /><button @click="log">打印日志</button>
</template>

三、计算属性computed

<script setup>
import { ref, computed } from 'vue'let count = ref(0)
const countPlus = computed(()=>{return count.value+1
})
</script><template><h1>计数:{{ countPlus }}</h1>
</template>

四、监听器watch、watchEffect

1、watch监听器除了使用方式有区别之外,其他的与vue2.0没啥变化

<script setup>
import { ref, reactive, watch } from 'vue'// 监听ref
let count = ref(0)
watch(count, (newVal, oldVal)=> {console.log('修改后', newVal)console.log('修改前', oldVal)
})// 监听reactive属性
const obj = reactive({count: 0
})
watch(()=> obj.count,     // 一个函数,返回监听属性(newVal, oldVal)=> {console.log('修改后', newVal)console.log('修改前', oldVal)},{immediate: true,     // 立即执行,默认为falsedeep: true     // 深度监听,默认为false}
)const onChange = function(){count.value++obj.count++
}
</script><template><button @click="onChange">改变count</button>
</template>

2、watchEffect

watchEffect是Vue3.0新增的一个监听属性的方法,它与watch的区别在于watchEffect不需要指定监听对象,回调函数里可直接获取到修改后的属性的值

<script setup>
import { ref, reactive, watchEffect } from 'vue'let count = ref(0)
const obj = reactive({count: 0
})
setTimeout(()=>{count.value++obj.count++
}, 1000)watchEffect(()=> {console.log('修改后的count', count.value)console.log('修改前的obj', obj.value)
})
</script>

五、自定义指令directive

vNameOfDirective的形式来命名本地自定义指令,可以直接在模板中使用

<script setup>
// 导入指令可重命名
// import { myDirective as vMyDirective } from './MyDirective.js'// 自定义指令
const vMyDirective = {beforeMount: (el) => {// 在元素上做些操作}
}
</script>
<template><h1 v-my-directive>This is a Heading</h1>
</template>

六、import导入的内容可直接使用

1、导入的模块内容,不需要通过methods来暴露它

// utils.js
export const onShow = function(name) {return 'my name is ' + name
}// Show.vue
<script setup>import { onShow } from './utils.js'
</script>
<template><div>{{ onShow('jack') }}</div>
</template>

2、导入外部组件,不需要通过components注册使用

// Child.vue
<template><div>I am a child</div>
</template>// Parent.vue
<script setup>import Child from './Child.vue'
</script>
<template><child></child>
</template>

七、声明props和emits

<script setup>中必须使用definePropsdefineEmitsAPI 来声明propsemits,它们具备完整的类型推断并且在<script setup>中是直接可用的

// Child.vue
<script setup>// 声明props
const props = defineProps({info: {type: String,default: ''}
})// 声明emits
const $emit = defineEmits(['change'])const onChange = function() {$emit('change', 'child返回值')
}
</script><template><h1>信息:{{ info }}</h1><button @click="onChange">点击我</button>
</template>// Parent.vue
<script setup>
import { ref } from 'vue'
import Child from './Child.vue'const msg = ref('hello setup !')    // 响应式变量const onAction = function(event) {console.log(event)    // child返回值
}
</script><template><child :info="msg" @change="onAction"></child>
</template>

如果使用了 Typescript,使用纯类型声明来声明 prop 和 emits也是可以的。

八、父组件获取子组件的数据

父组件要想通过ref获取子组件的变量或函数,子组件须使用defineExpose暴露出去

// Child.vue
<script setup>
import { ref, defineExpose } from 'vue'const info = ref('I am child')
const onChange = function() {console.log('Function of child')
}// 暴露属性
defineExpose({info,onChange
})
</script><template><h1>信息:{{ info }}</h1><button @click="onChange">点击我</button>
</template>// Parent.vue
<script setup>
import { ref } from 'vue'
import Child from './Child.vue'const childRef = ref()
const onAction = function() {console.log(childRef.value.info)    // I am childconsole.log(childRef.value.onChange())    // Function of child
}
</script><template><child ref="childRef"></child><button @click="onAction">获取子值</button>
</template>

九、provide和inject传值

无论组件层次结构有多深,父组件都可以通过**provide选项来其所有子组件提供数据,子组件通过inject**接收数据

// Parent.vue
<script setup>
import { ref, provide } from 'vue'
import Child from './Child.vue'const msg = ref('Hello, my son')
const onShow = function() {console.log('I am your parent')
}provide('myProvide', {msg,onShow
})
</script><template><child></child>
</template>// Child.vue
<script setup>
import { inject } from 'vue'const provideState = inject('myProvide')    // 接收参数const getData = function() {console.log(provideState.msg)    // Hello, my sonconsole.log(provideState.onShow())    // I am your parent
}
</script><template><button @click="getData">获取父值</button>
</template>

十、路由useRoute和useRouter

<script setup>
import { useRoute, useRouter } from 'vue-router'const $route = useRoute()
const $router = userRouter()// 路由信息
console.log($route.query)// 路由跳转
$router.push('/login')
</script>

十一、对await异步的支持

<script setup>中可以使用顶层await。结果代码会被编译成async setup()

<script setup>const post = await fetch(`/api/post/1`).then(r => r.json())
</script>

十二、nextTick

// 方式一
<script setup>
import { nextTick } from 'vue'nextTick(()=>{console.log('Dom已更新!')
})
</script>// 方式二
<script setup>
import { nextTick } from 'vue'await nextTick()    // nextTick是一个异步函数,返回一个Promise实例
// console.log('Dom已更新!')
</script>

十三、全局属性globalProperties

// main.js里定义
import { createApp } from 'vue'
import App from './App.vue'
const app = createApp(App)// 定义一个全局属性$global
app.config.globalProperties.$global = 'This is a global property.' app.mount('#app')// 组件内使用
<script setup>
import { getCurrentInstance } from 'vue'// 获取vue实例
const { proxy } = getCurrentInstance()
// 输出
console.log(proxy.$global)    // This is a global property.
</script>

十四、生命周期

setup()里访问组件的生命周期需要在生命周期钩子前加上“on”,并且没有beforeCreate和created生命周期钩子

因为setup是围绕beforeCreatecreated生命周期钩子运行的,所以不需要显式地定义它们。换句话说,在这些钩子中编写的任何代码都应该直接在setup函数中编写。

// 使用方式
<script setup>
import { onMounted } from 'vue'onMounted(()=> {console.log('onMounted')
})
</script>

十五、与普通的script标签一起使用

<script setup>可以和普通的<script>一起使用。普通的<script>在有这些需要的情况下或许会被使用到:

  • 无法在<script setup>声明的选项,例如inheritAttrs或通过插件启用的自定义的选项;

  • 声明命名导出,<script setup>定义的组件默认以组件文件的名称作为组件名;

  • 运行副作用或者创建只需要执行一次的对象。

十六、v-memo新指令

该指令与v-once类似,v-once是只渲染一次之后的更新不再渲染,而v-memo是根据条件来渲染。该指令接收一个固定长度的数组作为依赖值进行记忆比对,如果数组中的每个值都和上次渲染的时候相同,则该元素(含子元素)不刷新。

1、应用于普通元素或组件;

<template>
<-- 普通元素 -->
<div v-memo="[valueA, valueB]">...
</div><-- 组件 -->
<component v-memo="[valueA, valueB]"></component>
</template><script setup>
import component from "../compoents/component.vue"
</script>

当组件重新渲染的时候,如果valueAvalueB都维持不变,那么对这个<div>以及它的所有子节点的更新都将被跳过。

2、结合v-for使用

v-memo仅供性能敏感场景的针对性优化,会用到的场景应该很少。渲染v-for长列表 (长度大于 1000) 可能是它最有用的场景:

<template>
<div v-for="item in list" :key="item.id" v-memo="[item.id === selected]"><p>ID: {{ item.id }} - selected: {{ item.id === selected }}</p><p>...more child nodes</p>
</div>
</template>

当selected发生变化时,只有item.id===selected的该项重新渲染,其余不刷新。

style新特性

Vue3.2版本对单文件组件的style样式进行了很多升级,如局部样式、css变量以及样式暴露给模板使用等。

一、局部样式

<style>标签带有scopedattribute 的时候,它的 CSS 只会应用到当前组件的元素上:

<template><div class="example">hi</div>
</template><style scoped>
.example {color: red;
}
</style>

二、深度选择器

处于scoped样式中的选择器如果想要做更“深度”的选择,也即:影响到子组件,可以使用:deep()这个伪类:

<style scoped>
.a :deep(.b) {/* ... */
}
</style>

通过v-html创建的 DOM 内容不会被作用域样式影响,但你仍然可以使用深度选择器来设置其样式。

三、插槽选择器

默认情况下,作用域样式不会影响到<slot/>渲染出来的内容,因为它们被认为是父组件所持有并传递进来的。使用:slotted伪类以确切地将插槽内容作为选择器的目标:

<style scoped>
:slotted(div) {color: red;
}
</style>

四、全局选择器

如果想让其中一个样式规则应用到全局,比起另外创建一个<style>,可以使用:global伪类来实现:

<style scoped>
:global(.red) {color: red;
}
</style>

五、混合使用局部与全局样式

你也可以在同一个组件中同时包含作用域样式和非作用域样式:

<style>
/* global styles */
</style><style scoped>
/* local styles */
</style>

六、支持CSSModules

<style module>标签会被编译为CSS Modules并且将生成的 CSS 类键暴露给组件:

1、 默认以$style对象暴露给组件;

<template><p :class="$style.red">This should be red</p>
</template><style module>
.red {color: red;
}
</style>

2、可以自定义注入module名称

<template><p :class="classes.red">red</p>
</template><style module="classes">
.red {color: red;
}
</style>

七、与setup一同使用

注入的类可以通过useCssModuleAPI 在setup()<script setup>中使用:

<script setup>
import { useCssModule } from 'vue'// 默认, 返回 <style module> 中的类
const defaultStyle = useCssModule()// 命名, 返回 <style module="classes"> 中的类
const classesStyle = useCssModule('classes')
</script>

八、动态 CSS

单文件组件的<style>标签可以通过v-bind这一 CSS 函数将 CSS 的值关联到动态的组件状态上:

<script setup>
const theme = {color: 'red'
}
</script><template><p>hello</p>
</template><style scoped>
p {color: v-bind('theme.color');
}
</style>

(完)

参考文献:

SFC 语法规范 | Vue.js

Vue3.2 setup语法糖、Composition API、状态库Pinia归纳总监

先自我介绍一下,小编13年上师交大毕业,曾经在小公司待过,去过华为OPPO等大厂,18年进入阿里,直到现在。深知大多数初中级java工程师,想要升技能,往往是需要自己摸索成长或是报班学习,但对于培训机构动则近万元的学费,着实压力不小。自己不成体系的自学效率很低又漫长,而且容易碰到天花板技术停止不前。因此我收集了一份《java开发全套学习资料》送给大家,初衷也很简单,就是希望帮助到想自学又不知道该从何学起的朋友,同时减轻大家的负担。添加下方名片,即可获取全套学习资料哦

Vue3.2单文件组件setup的语法糖总结相关推荐

  1. 使用vue-cli创建Vue工程化项目及单文件组件的创建和调用

    文章目录 1. Vue 单文件组件的优势 2. cli 创建 Vue 工程项目 3. 单文件组件的创建和调用 1. Vue 单文件组件的优势 把一个组件全部内容汇合到一个文件中,文件名字是以 .vue ...

  2. vue3 单文件组件

    1.使用setup组件自动注册 在 script setup 中,引入的组件可以直接使用,无需再通过components进行注册,并且无法指定当前组件的名字,它会自动以文件名为主,也就是不用再写nam ...

  3. 丁鹿学堂:前端进阶学习vue3最新教程之vue的单文件组件深入理解

    在vue3中,template模版中的html代码,并不能直接在网页中执行,而是会在项目运行时,在浏览器中被编译为js 的函数去执行. 问题:在template模版中写html,体验很差,智能提示都不 ...

  4. Vue第七章:项目环境配置及单文件组件 vue脚手

    第七章:项目环境配置及单文件组件 vue脚手架 回顾: 组件之间的通信 父传子:正向传递 vue允许 自动触发 ​ props ​ 1.先在子组件中定义期待的属性名和类型 ​ 2.在父组件中调用子组件 ...

  5. Vue单文件组件基础模板

    背景 相信大家在使用Vue开发项目时,基本都是以单文件组件的形式开发组件的,这种方式好处多多: 1.代码集中,便于开发.管理和维护 2.可复用性高,直接将vue文件拷贝到新项目中 我暂时就想到这两点, ...

  6. Vue 单文件组件||Vue 单文件组件的基本用法||webpack 中配置 vue 组件的加载器|| 在 webpack 项目中使用 vue

    Vue 单文件组件 传统组件的问题和解决方案 1. 问题 1. 全局定义的组件必须保证组件的名称不重复 2. 字符串模板缺乏语法高亮,在 HTML 有多行的时候,需要用到丑陋的 \ 3. 不支持 CS ...

  7. Vue.js 单文件组件

    单文件组件 介绍 在很多 Vue 项目中,我们使用 Vue.component 来定义全局组件,紧接着用 new Vue({ el: '#container '}) 在每个页面内指定一个容器元素. 这 ...

  8. Vue单文件组件与vue-loader

    单文件组件 Vue.js是一个渐进式的js框架,在使用wewbpack构建Vue项目时,可以使用一种新的构建模式: .vue单文件组件. 就是一个后缀名为.vue的文件,在webpack中使用vue- ...

  9. vue iframe 中写script_vue: 单文件组件 render函数

    使用vue-cli创建的vue项目,如何在这种项目中使用组建? 首先创建项目.启动项目 我们再来了解一下目录结构,src文件夹是写逻辑代码的地方,public是最终渲染到浏览器的地方. 在public ...

最新文章

  1. h5引入不同的js文件怎样让第二个js使用第一个js文件中的函数_px2rem-loader使用及注意事项...
  2. 算法笔记-图--bfs
  3. ViewPager+Fragment切换时无法更新数据问题解析(源代码分享)
  4. PyQt5简介及demo
  5. Win64 驱动内核编程-11.回调监控进线程句柄操作
  6. Java JSON 之 Xml 转 JSON 字符串
  7. mysql 查询姓张或者姓王_mysql查询练习
  8. 80x86 CPU 的工作模式
  9. JS Ajax异步请求发送列表数据后面多了[]
  10. server.mappath php,Server.MapPath( ) 方法的主要功能是获取文件的绝对路径。
  11. toj 4315 一二三
  12. Exchange 2010 SP2 新功能
  13. 《淘宝网开店 拍摄 修图 设计 装修 实战150招》一一2.8  黄金分割的三分法构图...
  14. weui 加载提示_WeUI与WeUI.JS配合切换进入页面显示加载动画
  15. 心跳检测的思路及代码
  16. UCF Local Programming Contest 2018题解
  17. java 使用adobe fms流媒体
  18. C++中的随机数函数(
  19. 超弦理论、M理论和量子力学
  20. 毕业论文浅析计算机病毒,浅析计算机病毒的有效防御论文

热门文章

  1. jsp+ssm计算机毕业设计宾馆客房管理系统【附源码】
  2. 【Linux】了解根目录下每个文件的作用
  3. 开源导航软件Navit介绍
  4. [附源码]计算机毕业设计JAVA游戏战队考核系统
  5. 计算机入门及操作技能训练,计算机入门及操作技能训练.ppt
  6. 致敬2202年,这些优秀的裁缝们
  7. 前端画图之iphoneSE主屏
  8. 小学计算机ps课题计划,PS教学授课进度的计划表.doc
  9. 第二十三模板 18.4算法类
  10. Romberg(龙贝格)积分法