Vue2入门笔记(2) | 总字数: 3.6k | 阅读时长: 14分钟 | 浏览量: |
基于Bilibili黑马程序员Vue2+vue3 教程的学习笔记(2)
Vue-cli 什么是单页面应用程序,SPA(Single Page Application)
所有的功能与交互都在这唯一的一个页面内完成。
vue-cli是Vue.js开发的标准工具。简化了程序员基于webpack创建工程化的Vue项目的工程。
中文官网
安装和使用 它是npm上的一个全局包使用npm install -g @vue/cli
来进行安装,安装完成后使用vue --version
来查看是否安装成功
创建项目
会出现如下提示,并选择手动
1 2 3 4 5 6 7 8 Vue CLI v5.0.8 ? Please pick a preset: > Default ([Vue 3] babel, eslint) Default ([Vue 2] babel, eslint) Manually select features
使用空格选择与取消部分选项,然后回车
选择vue2.x,然后回车
选择Less,回车
选择第一项,将babel等插件的配置项,放到自己独立的文件中,回车
输入y保存我们的选项作为预设,回车
输入为预设取的别名,然后回车
等待加载完成后,进入项目目录,执行npm run serve
命令让项目跑起来,打开控制台打印的url地址,查看效果。
src目录和构成 1 2 3 4 assets文件夹:存放项目中用到的静态资源文件,例如:css样式表、图片资源 components文件夹:程序员封装的、可复用的组件,都要放到componts目录下 main.js文件:项目的入口,整个项目的运行,要先执行main.js App.vue:是项目的根组件
vue项目的运行流程 通过main.js把App.vue渲染到index.html的指定区域中
App.vue 用来编写待渲染的模板结构 index.html中需要预留一个el区域 mian.js把App.vue渲染到了index.html所预留的区域中 1 2 3 4 5 6 7 8 9 10 import Vue from 'vue' import App from './App.vue' Vue .config .productionTip = false new Vue ({ render : h => h (App ), }).$mount('#app' )
vue组件 只要是.vue
结尾的文件,就是vue组件
vue组件的三个组成部分 template 组件的模板结构 script 组件的JavaScript行为 style 组件的样式 script
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 <template> <div class="test-box"> <h3>这是用户自定义的Test.vue---{{ username }}</h3> <button @click="changeName">修改用户名</button> </div> </template> <script> // 默认导出,这是固定写法 export default { // .vue组件中的data不能像之前一样,不能指向对象。 // 组件中的data必须是一个函数 /*data: { username: '张三' }*/ data() { // 这个return出去的对象中,可以定义数据 return { username: '张三' } }, method: { changeName() { // 在组件中,this就表示当前组件的实例对象 console.log(this); this.username = '李四'; } }, // 当前组件中的侦听器 watch: { }, // 当前组件中的计算属性 computed: { }, // 当前组件的过滤器 filters: { } } </script> <style> .test-box { background-color: pink; } </style>
在template节点下只能有一个根元素
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 <!-- 错误示范 --> <template> <div class="test-box"> <h3>这是用户自定义的Test.vue---{{ username }}</h3> <button @click="changeName">修改用户名</button> </div> <div> </div> </template> <!-- 正确示范 --> <template> <div> <div class="test-box"> <h3>这是用户自定义的Test.vue---{{ username }}</h3> <button @click="changeName">修改用户名</button> </div> <div> </div> </div> </template>
在style节点支持less语法
1 2 3 4 5 6 7 <!-- 支持less --> <style lang="less"> </style> <!-- 不支持less,支持css,可不写 --> <style lang="css"> </style>
组件之间的父子关系 组件封装好之后,彼此之间是相互独立的,不存在父子关系
在使用组件时,根据彼此的嵌套关系,形成了父子关系,兄弟关系
使用组件的三个步骤 使用import语法导入需要的组件 使用components节点注册组件 以标签的形式使用刚才注册的组件 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 <template> <div class="box"> <!-- 以标签形式,使用注册好的组件 --> <Left></Left> </div> </template> <script> // 导入需要使用的.vue组件 import Left from '@/components/left.vue'; import Right from '@/components/right.vue'; export default { components: { 'Left': Left, 'Right': Right } } </script>
visual studio code 路径提示插件Path Autocomplete
注册全局组件 之前使用components注册的是私有组件,只能在被导入的组件里使用
在vue项目的main.js 入口文件中,通过Vue.component()
方法,可以注册全局组件
1 2 3 4 5 import Count from '@/components/Count.vue' Vue .componnent ('MyCount' ,Count )
Visual Studio Code自动补齐标签插件Auto Close Tag
组件里的props 组件的自定义属性,在封装通用组件的时候,合理使用props可以极大提高组件的复用性
封装者通过props可以让使用者在使用组件时传参
props中的数据,可以直接在模板结构中使用,不能直接修改props的值,props中的值是只读的(可写,但是会报错)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 export default { props : ['init' ] data ( ) { return { count : this .init } } }
props default默认值
1 2 3 4 5 6 7 8 9 10 11 12 export default { props : { init : { default : 0 } } }
props type值类型
1 2 3 4 5 6 7 8 9 10 11 12 13 14 export default { props : { init : { default : 0 , type : Number } } }
props required必填项
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 export default { props : { init : { default : 0 , type : Number , required : true } } }
样式冲突 写在.vue组件中的样式会全局生效,会造成多个组件样式冲突的问题。
为组件中的style标签加上scoped
属性,可以解决这个问题。
scoped原理:为组件中每个标签添加一个自定义属性,在style中使用属性选择器。
1 2 3 4 5 <style lang="less" scoped> h3 { color: red; } </style>
如果使用scoped
属性,那么它的父元素就不能修改它的样式,就需要用到/deep/
修改子组件的样式
当使用第三方组件库的时候,如果需要更改第三方默认样式,则需要用到/deep/
1 2 3 4 5 <style lang="less" scoped> /deep/ h5 { color: red; } </style>
vue组件的实例对象 浏览器并不能识别.vue文件,是由vue-template-compiler
包编译成JS交给浏览器来执行。
以标签形式引用组件才会创建组件实例。
生命周期 一个组件从创建 -> 运行 -> 销毁的整个阶段,强调的是一个时间段。
vue2官方文档
一使用组件的标签就相当于new了一个组件实例
组件创建阶段
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 export default { data ( ) { return { message : 'hello' , } }, beforeCreate ( ) { }, created ( ) { concole.log (this .message ); }, beforMount ( ) { }, mounted ( ) { }, beforeUpdate ( ) { }, updated ( ) { }, beforeDestroy ( ) { } }
created生命周期函数非常常用,经常在它里面调用methods中的方法,请求服务器的数据,并且将请求到的数据,转存到data中,供template模板渲染使用
组件运行阶段
组件销毁阶段
父子传值 父向子使用自定义属性 父类在标签中使用props传值+标签传值,子组件定义自定义属性
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 <!--子组件中定义 --> <script> export default { props: ['init'] } </script> <!-- 父组件中定义 --> <template> <Son :init="msg"></Son> </template> <script> export default { data() { return { msg: 'hello' } } } </script>
不要修改props的值
子向父使用自定义事件 子向父共享数据使用自定义事件
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 <!-- 子组件 --> <script> export default { data() { return { count: 0 } }, methods: { add() { this.count += 1; // 修改数据时,通过$emit()触发自定义事件 this.$emit('numchange', this.count); } } } </script> <!-- 父组件 --> <template> <!-- 绑定自定义事件 --> <Son @numchange="getNumCount"></Son> </template> <script> export default { data() { return { countFromSon: 0 } }, methods: { getNewCount(val) { this.countFromSon = val } } } </script>
兄弟组件之间的数据共享 Vue2中,兄弟组件之间数据共享的方案是EventBus
这是一种现成的解决方案
使用步骤
创建eventBus.js模块,并向外共享一个Vue的实例对象 在数据发送方,调用bus.$emit('事件名称',要发送的数据)
方法触发自定义事件 在数据接收方,调用bus.$('事件名称',事件处理函数)
方法注册一个自定义事件 ref引用 在vue中,程序员不需要操作DOM,只需要将数据维护好。(数据驱动视图)
在vue项目中,强烈不建议安装和使用jQuery
ref辅助程序员在不依赖jQuery的情况下,获取DOM元素或组件的引用
每个vue的组件实例上,都包含一个$refs
对象,里面存储看对应的DOM元素或组件的引用。默认情况下,组件的$refs
指向一个空对象。
ref引用DOM 如何拿到某个DOM元素的引用
为标签添加ref
属性,一个组件里属性的值不能重复 使用$refs
对象获取标签指定的ref
值 1 2 3 4 5 6 7 8 9 10 11 <template> <h5 ref="myTag"></h5> </template> <script> export default { created() { console.log(this.$refs.myTag); } } </script>
ref应用组件 ref也可以用在组件的标签引用上,用$refs
访问后是组件的实例对象,就可以访问它的属性和方法
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 <template> <h5 ref="myTag"></h5> <MyComp ref="MyComp"></MyComp> </template> <script> import MyComp from '@/components/MyComp.vue' export default { created() { console.log(this.$refs.myTag); }, components: { MyComp }, created() { console.log(this.$refs.myTag); } } </script>
this.$nextTick(callback)
将callback函数在页面重新渲染好后再执行。
因为有时候页面还没更新就获取ref
的值,那个时候ref
还并没有出现,所以需要用到this.$nextTick(callback)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 <template> <h5 v-if="isShow" ref="myTag"></h5> <h2 v-else ref="myTag2"></h2> <button @click="doSome">显示h2</button> </template> <script> export default { data() { return { isShow: true } } methods: { doSome() { this.isShow = false; // 因为数据更新了,但是DOM还没有重新加载,所以要使用$nextTick方法 this.$nextTick(() => { console.log(this.$refs.myTag) }) } } } </script>
数组中的方法 some循环 forEach循环一旦开始就无法在中间停止
1 2 3 4 5 6 7 8 9 const arr = ['a' ,'b' ,'c' ,'d' ,'e' ,'f' ]arr.some ((item, index ) => { if (item === c) { console .log (item) return true } })
every循环 能用来判断数组中每个对象的某个属性,例如判断是否被全选
1 2 3 4 5 6 7 8 9 10 const arr = [ {id : 1 , name : '香蕉' , state : true }, {id : 2 , name : '苹果' , state : true }, {id : 3 , name : '梨子' , state : true }, {id : 4 , name : '橘子' , state : true } ] const result = arr.every (item => item.state )console .log (result)
reduce基本使用 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 const arr = [ {id : 1 , name : '香蕉' , state : true , price : 10 , count : 1 }, {id : 2 , name : '苹果' , state : false , price : 80 , count : 2 }, {id : 3 , name : '梨子' , state : false , price : 22 , count : 2 }, {id : 4 , name : '橘子' , state : true , price : 24 , count : 5 } ] let amtconst result = arr.filter (item => item.state ).reduce ((amt, item ) => { return amt += item.price * item.count },0 ) console .log (result)
reduce简化写法 1 2 3 let amtconst result = arr.filter (item => item.state ).reduce ((amt, item ) => amt += item.price * item.count ,0 )console .log (result)