# vue_basic
**Repository Path**: yanjiushen/vue_basic
## Basic Information
- **Project Name**: vue_basic
- **Description**: Vue基础教程
- **Primary Language**: Unknown
- **License**: Not specified
- **Default Branch**: master
- **Homepage**: None
- **GVP Project**: No
## Statistics
- **Stars**: 0
- **Forks**: 1
- **Created**: 2021-12-19
- **Last Updated**: 2022-03-21
## Categories & Tags
**Categories**: Uncategorized
**Tags**: Vue, 基础教程
## README
# Vue 基础
特点:组件化模式、声明式编码、使用虚拟 DOM+优秀的 diff 算法
## 1.初识 Vue
```js
Vue.config.productionTip = false; // 关闭Vue生产提示
```
1. 创建 Vue 实例
```js
new Vue({
el: "#root", //el用于指定当前Vue实例为哪个容器服务,值通常为css选择器字符串
data: {
// data中用于存储数据,数据供el所指定的容器去使用
name: "尚硅谷",
},
});
```
2. 总结
1. 想让 Vue 工作,就必须创建一个 Vue 实例,且要传入一个配置对象;
2. root 容器里的代码依然符合 html 规范,只不过混入了一些特殊的 Vue 语法;
3. root 容器里的代码被称为【Vue 模板】;
4. Vue 实例和容器是一一对应的;
5. 真实开发中只有一个 Vue 实例,并且会配合着组件一起使用;
6. {{xxx}}中的 xxx 要写 js 表达式,且 xxx 可以自动读取到 data 中的所有属性;
7. 一旦 data 中的数据发生改变,那么模板中用到该数据的地方也会自动更新;
```
注意区分:js表达式 和 js代码(语句)
1. 表达式:一个表达式会产生一个值,可以放在任何一个需要值的地方:
(1). a
(2). a+b
(3). demo(1)
(4). x === y ? 'a' : 'b'
2. js代码(语句)
(1). if(){}
(2). for(){}
```
## 2.模板语法
Vue模板语法有2大类:
1. 插值语法:
功能:用于解析标签体内容。
写法:{{xxx}},xxx是js表达式,且可以直接读取到data中的所有属性。
2. 指令语法:
功能:用于解析标签(包括:标签属性、标签体内容、绑定事件......)。
举例:v-bind:href="xxx" 或 简写为 :href="xxx",xxx同样要写表达式,且可以直接读取到data中的所有属性。
备注:Vue中有很多的指令,且形式都是 v-xxx
## 3.数据绑定
```
单向数据绑定:
双向数据绑定:
单向数据绑定:
双向数据绑定:
```
> `v-model` 只能应用在表单类元素中(输入类元素)
总结:
Vue 中有两种数据绑定的方式:
1. 单向绑定(`v-bind`):数据值能从 data 流向页面。
2. 双向绑定(`v-model`):数据不仅能从 data 流向页面,还可以从页面流向 data。
备注:
- 双向绑定一般都应用在表单类元素上(如:input、select 等)
- v-model:value 可以简写为 v-model,因为 v-model 默认手机的就是 value 值。
## 4.el 与 data 的两种写法
1. el 有两种写法
(1). new Vue 时配置 el 属性。
```js
const vm = new Vue({
el: "#root", // 第一种写法
});
```
(2). 先创建 Vue 实例,随后再通过 vm.$mount('#root')指定 el 的值。
```js
const vm = new Vue({});
vm.$mount("#root"); // 第二种写法
```
2. data 有两种写法
(1). 对象式
```js
new Vue({
el: "#root",
// data的第一种写法:对象式
data: {
name: "尚硅谷",
},
});
```
(2). 函数式
```js
new Vue({
el: "#root",
data() {
return {
name: "尚硅谷",
};
},
});
```
> 如何选择:目前哪种写法都可以,以后学习到组件时,data 必须使用函数式,否则会报错。
3. 一个重要的原则;
- 由 Vue 管理的函数,一定不要写箭头函数,一旦写了箭头函数,this 就不再是 Vue 实例了。
## 5.MVVM 模型
1. M:模型(Model):data 中的数据
2. V:视图(View):模板代码
3. VM:视图模型(ViewModel):Vue 实例
观察发现:
1. data 中所有的属性,最后都出现在了 vm 身上。
2. vm 身上所有的属性及 Vue 原型上所有属性,在 Vue 模板中都可以直接使用。
## 6.数据代理
1. 回顾 Object.defineProperty
```js
let number = 3;
let person = {
name: "张三",
sex: "男",
// age: 18
};
// console.log(Object.keys(person)) // name sex age
// 添加的属性默认不可被枚举
Object.defineProperty(person, "age", {
// value: 18,
// enumerable: true, // 控制属性是否可以枚举 默认:false
// writable: true, // 控制属性是否可以被修改 默认:false
// configurable: true, // 控制属性是否可以被删除 默认:false
// 当有人读取person的age属性时,get函数就会被调用,且返回值就是age的值
get() {
console.log("有人读取age属性了");
return number;
},
// 当有人修改person的age属性时,set函数(setter)就会被调用,且会收到修改的具体值
set(value) {
console.log("有人修改了age属性,且值是:", value);
number = value;
},
});
console.log(Object.keys(person)); // name sex
console.log(person);
```
2. 数据代理
```js
let obj = { x: 100 };
let obj2 = { y: 200 };
Object.defineProperty(obj2, "x", {
get() {
return obj.x;
},
set(value) {
obj.x = value;
},
});
```
> 数据代理,通过一个对象代理对另一个对象中属性的操作(读/写)
3. Vue 中的数据代理
(1). Vue 中的数据代理:
通过 vm 对象来代理 data 对象中属性的操作(读/写)
(2). Vue 中数据代理的好处:
更加方便的操作 data 中的数据
(3). 基本原理
通过 Object.defineProperty()把 data 对象中所有属性添加到 vm 上。
在 getter/setter 内部去操作(读/写)data 中对应的属性。

## 7. 事件的基本使用
> 1. 使用 v-on:xxx 或 @xxx 绑定事件,其中 xxx 是事件名;
> 2. 事件的回调需要配置在 methods 对象中,最终会在 vm 上;
> 3. methods 中配置的函数,不要用箭头函数!否则 this 就不是 vm 了;
> 4. methods 中配置的函数,都是被 Vue 所管理的函数,this 的指向是 vm 或 组件实例对象;
> 5. @click="demo" 和 @click="demo($event)" 效果一致,但后者可以传参。
## 8. 事件修饰符
Vue 的事件修饰符:
1. prevent:阻止默认事件(常用);(e.preventDefault())
2. stop:阻止事件冒泡(常用);(e.stopPropagation())
3. once:事件只触发一次(常用);
4. capture:使用事件的 捕获模式;
5. self:只有 event.target 是当前操作的元素时才出发事件;
6. passive:事件的默认行为立即执行,无需等待事件回调执行完毕。
```
```
## 9.键盘事件
```html
Vue.config.keyCodes.huiche = 13
console.log(e.key) // 按键名字 console.log(e.keyCode) // 按键编码
console.log(e.target.value)
```
总结:
1. Vue 中常用的按键别名:
回车 => enter
删除 => delete (捕获“删除”和“退格”键)
退出 => esc
空格 => space
换行 => tab (特殊,必须配合 keydown 去使用)
上 => up
下 => down
左 => left
右 => right
2. Vue 未提供别名的按键,可以使用按键原始的值去绑定,但注意要转为 kebab-case(短横线命名)。
3. 系统修饰键(用法特殊):`ctrl、alt、shift、meta`
(1). 配合 keyup 使用:按下修饰键的同时,再按下其他键,随后释放其他键,事件才能被触发。
(2). 配合 keydoem 使用:正常触发事件。
4. 也可以使用 keyCode 去指定具体的按键(不推荐)
5. `Vue.config.keyCodes.自定义键名 = 键码`,可以去定制按键别名
## 10.计算属性
1. 定义:要用的属性不存在,要通过已有属性计算得来。
2. 原理:底层借助了 Object.defineProperty 方法提供的 getter 和 setter。
3. get 函数什么时候执行?
(1). 初次读取时会执行一次。
(2). 当依赖的数据发生改变时会被再次调用。
4. 优势:与 methods 实现相比,内部有缓存机制(复用),效率更高,调试方便。
5. 备注:
(1). 计算属性最终会出现在 vm 上,直接读取使用即可。
(2). 如果计算属性要被修改,那必须写 set 函数去响应修改,且 set 中要引起计算时依赖的数据发生改变。
```js
computed: {
fullName: {
get() {
return this.firstName +'-'+ this.lastName
},
set(value) {
this.firstName = value.split('-')[0]
this.lastName = value.split('-')[1]
}
},
// 简写
fullName() {
return this.firstName +'-'+ this.lastName
}
}
```
## 11.监视属性 watch
1. 当被监视的属性变化时,回调函数自动调用,进行相关操作
2. 监视的属性必须存在,才能进行监视
3. 监视的两种写法:
(1). new Vue 时传入 watch 配置
(2). 通过 vm.$watch 监视
```js
// watch: {
// isHot: {
// // 初始化时让handler调用一下
// immediate: true,
// // handler什么时候调用?当isHot发生改变时
// handler(newValue, oldValue) {
// console.log("newValue:",newValue,",oldValue:", oldValue)
// }
// }
// }
```
## 12.深度监视
1. Vue 中的 watch 默认不监测对象内部值的改变(一层)。
2. 配置 `deep:true` 可以监测对象内部值改变(多层)。
备注:
(1). Vue 自身可以监测对象内部值的改变,但 Vue 提供的 watch 默认不可以!
(2). 使用 watch 时根据数据的具体结构,决定是否采用深度监视。
## 13.computed 和 watch 之间的区别
1. computed 能完成的功能,watch 都可以完成。
2. watch 能完成的功能,computed 不一定能完成,例如:`watch 可以进行异步操作`。
两个重要的小原则:
1. `所有被 Vue 管理的函数,最好写成普通函数`,这样 this 的指向才是 vm 或 组件实例对象。
2. 所有不被 Vue 管理的函数(定时器的回调函数、ajax 的回调函数、Promise 的回调函数等),最好写成箭头函数,这样 this 的指向才是 vm 或 组件实例对象。
## 14.绑定样式
1. class 样式
写法 `:class="xxx"` xxx 可以时字符串、对象、数组。
字符串写法适用于:类名不确定,要动态获取。
对象写法适用于:要绑定多个样式,个数不确定,名字也不确定。
数组写法适用于:要绑定多个样式,个数确定,名字也确定,但不确定用不用。
2. style 样式
`:style="{fontSize: xxx}"`其中 xxx 是动态值。
`:style="[a, b]"`其中 a、b 是样式对象。
## 15.条件渲染
1. `v-if`
写法:
(1). v-if="表达式"
(2). v-else-if="表达式"
(3). v-else="表达式"
适用于:切换频率`较低`的场景。
特点:不展示的 DOM 元素直接被删除。
注意:`v-if `可以和:`v-else-if`、`v-else` 一起使用,但要求结构`不能被“打断”`。
2. `v-show`
写法:v-show="表达式"
适用于:切换频率`较高`的场景。
特点:不展示的 DOM 元素不被移除,仅仅是使用样式隐蔽掉。
3. 备注:使用 v-if 时,元素可能无法被获取到,而使用 v-show 一定可以被获取到。
## 16.v-for 指令
1. 用于展示列表数据
2. 语法:v-for="(item, index) in xxx" :key="yyy"
3. 可遍历:数组、对象、字符串(用的很少)、指定次数(用的很少)
## 17.面试题:react、Vue 中的 key 有什么作用?(key 的内部原理)
1. 虚拟 DOM 中 key 的作用:
key 是虚拟 DOM 对象的标识,当数据发生变化是,Vue 会根据【新数据】生成【新的虚拟 DOM】,随后 Vue 进行【新虚拟 DOM】与【旧虚拟 DOM】的差异比较,比较规则如下:
2. 对比规则:
(1). 旧虚拟 DOM 中找到了与新虚拟 DOM 相同的 key:
- 若虚拟 DOM 中内容没变,直接使用之前的真实 DOM!
- 若虚拟 DOM 中内容变了,则生成新的真实 DOM,随后替换掉页面中之前的真实 DOM。
(2). 旧虚拟 DOM 中未找到与新虚拟 DOM 相同的 key
- 创建新的真实 DOM,随后渲染到新页面。
3. 用 index 作为 key 可能会引发新的问题:
1. 若对数据进行:逆序添加、逆序删除等破坏顺序操作:
- 会产生没有必要的真实 DOM 更新 ==> 界面效果没问题,但效率低。
2. 如果结构中还包含输入类的 DOM:
- 会产生错误 DOM 更新 ==> 界面有问题。
4. 开发中如何选择 key?
1. 最好使用每条数据的唯一标识作为 key,比如 id、手机号、身份证号、学号等唯一值。
2. 如果不存在对数据的逆序添加、逆序删除等破坏性操作,仅用于渲染列表用于展示,使用 index 作为 key 是没有问题的。
## 18.Vue 监视数据的原理
1. vue 会监视 data 中`所有层次`的数据。
2. 如何监视对象中的数据?
通过`setter`实现监视,且要在`new Vue`时就传入 要监视的数据。
(1). 对象中后追加的属性,Vue 默认不做响应式处理
(2). 如需给后添加的属性做响应式,请使用如下 API:
> Vue.set(target, propertyName/index, value) 或
> vm.$set(target, propertyName/index, value)
3. 如何监测数组中的数据?
通过包裹数组更新元素的方法实现,本质就是做了两件事:
(1). 调用原生对应的方法对数组进行更新。
(2). 重新解析模板,进而更新页面。
4. 在 Vue 修改数组中的某个元素一定要用如下方法:
1. 使用这些 API:`push()`、`pop()`、`shift()`、`unshift()`、`splice()`、`sort()`、`reverse()`
2. `Vue.set()` 或 `vm.$set()`
特别注意:`Vue.set()` 和 `vm.$set()` 不能给 vm 或 vm 的根实例对象添加属性!!!
```js
addSexType(){ //添加性别属性 默认值:男
// Vue.set(this.student,'sex','男')
this.$set(this.student,'sex','男')
},
addFriend(){ //添加一个朋友
this.student.friend.unshift({name:'tony',age:'25'})
},
editFirstFriendName(){ //修改第一个朋友的名字为张三
// this.student.friend[0].name = '张三'
// Vue.set(this.student.friend[0],'name','张三')
this.$set(this.student.friend[0],'name','张三')
},
addHobby(){ //添加一个爱好
this.student.hobby.push("搓背")
},
editFirstHobby(){ //修改第一个爱好
// this.student.hobby.splice(0,1,'开车')
// Vue.set(this.student.hobby,0,'开车')
this.$set(this.student.hobby,0,'开车')
},
filHobby(){ //过滤掉爱好中的抽烟
this.student.hobby = this.student.hobby.filter((e) => {
return e != '抽烟'
})
}
```
## 19.收集表单数据
若: ``,则`v-model`收集的是`value`值,用户输入的就是`value`值。
若:``,则`v-model`收集的是`value`值,且要给标签配置`value`值。
若:``
1. 没有配置 input 的 value 属性,那么收集的就是 checked(勾选 or 未勾选,是布尔值)
2. 配置 input 的 value 属性:
(1). v-model 的初始值是非数组,那么收集的就是 checked(勾选 or 未勾选,是布尔值)
(2). v-model 的初始值是数组,那么收集的就是 value 组成的数组
备注:v-model 的三个修饰符:
- lazy:失去焦点再收集数据
- number:输入字符串转为有效的数字
- trim:输入首尾空格过滤
## 20.过滤器
定义:对要显示的数据进行特定 格式化后再显示(使用与一些简单的逻辑的处理)。
语法:
1. 注册过滤器:`Vue.filter(name, callback)` 或 `new Vue({filters:{}})`
2. 使用过滤器,`{{ xxx | 过滤器名}}` 或 `v-bind:属性 = "xxx | 过滤器名"`
备注:
1. 过滤器也可以接收额外参数,多个过滤器也可以串联
2. 并没有改变原本的数据,是产生新的对应的数据
```js
当前时间:{{time | formatTime}}
当前时间:{{time | formatTime('YYYY_MM_DD')}}
当前时间:{{time | formatTime | mySlice}}
{{name | mySlice}}
ss
// 全局过滤器
Vue.filter("mySlice", function (value) {
return value.slice(0, 4);
});
// 局部过滤器
new Vue({
filters: {
formatTime(time, str = "YYYY-MM-DD HH:mm:ss") {
return dayjs(time).format(str);
},
},
});
```
## 21.内置指令
`v-bind`:单向绑定解析表达式,可简写为 `:xxx`
`v-model`:双向数据绑定
`v-for`:遍历数组/对象/字符串
`v-on`:绑定事件监听,可简写为`@`
`v-if`:条件渲染(动态控制节点是否存在)
`v-else`:条件渲染(动态控制节点是否存在)
`v-show`:条件渲染(动态控制节点是否展示)
`v-text`:
1. 作用:向其所在的节点中渲染文本内容。
2. 与插值语法的区别:`v-text` 会替换点节点中的内容,`{{xxx}}`则不会
`v-html`:
1. 作用:向指定节点中渲染包含 html 结构的内容。
2. 与插值语法的区别:
(1). v-html 会替换掉节点中所有的内容,{{xxx}}则不会。
(2). v-html 可以识别 html 结构。
3. 严重注意:v-html 有安全问题!!!
(1). 在网站上动态渲染任意 HTML 是非常危险的,容易导致 XSS 攻击。
(2). 一定要在可信的内容上使用 v-html,永远不要用在用户提交的内容上!
```js
;
str2: '这里有你要的资源';
```
`v-cloak`(没有值):
1. 本质是一个特殊属性,Vue 实例创建完毕并接管容器后,会删掉 v-cloak 属性。
2. 使用 css 配合 v-cloak 可以解决网速慢时页面展示出{{xxx}}的问题。
```js
[v-cloak] {
display: none;
}
{{name}}
```
`v-once`:
1. v-once 所在节点在初次动态渲染后,就视为静态内容了。
2. 以后数据的改变不会引起 v-once 所造结构的更新,可以用于优化性能。
```js
初始化的n值是:{{n}}
当前的n值是:{{n}}
```
`v-pre`:
1. 跳过其所在节点的编译过程。
2. 可利用它跳过:没有使用指令语法、没有使用插值语法的节点,会加快编译。
```html
没有使用指令语法
当前的n值:{{n}}
```
## 22.自定义指令
1. 定义语法:
(1). 局部指令:
```js
new Vue({ new Vue({
directives:{ directives:{
指令名: 配置对象 或 指令名: 回调函数
} }
}) })
```
(2). 全局指令:
`Vue.directive(指令名, 配置对象)` 或 `Vue.directive(指令名, 回调函数)`
2. 配置对象中常用的 3 个问题:
(1). `bind`:指令与元素成功绑定是调用、
(2). `inserted`:指令所在元素被插入页面时调用。
(3). `update`:指令所在模板结构被重新解析时调用。
3. 备注:
1. 指令定义时不加 v-,但使用时要加 v-;
2. 指令名如果是多个单词,要使用 kebab-case 命名方式,不要用 camelCase 命名。
```js
new Vue({
directives: {
// big函数何时会被调用:1.指令与元素成功绑定时 2.指令所在的模板被重新解析时
big(element, binding) {
element.innerText = binding.value * 10;
},
fbind: {
// 指令与元素成功绑定时(一上来)
bind(element, binding) {
console.log(this); // 此处的this是window
element.value = binding.value;
},
// 指令所在元素被插入页面时
inserted(element, binding) {
element.focus();
},
// 指令所在的模板被重新解析时
update(element, binding) {
element.value = binding.value;
},
},
},
});
```
## 23.生命周期
1. 又名:生命周期回调函数、生命周期函数、生命周期钩子。
2. 是什么:Vue 在关键时刻帮我们调用的一些特殊名称的函数。
3. 生命周期函数的`名字不可更改`,但函数的具体内容是程序员根据需求编写的。
4. 生命周期函数中的 this 指向是 vm 或 组件实例对象。
常用的生命周期钩子:
1. `mounted`:发送 ajax 请求、启动定时器、绑定自定义事件、订阅消息等【初始化操作】。
2. `beforeDestory`:清除定时器、解绑自定义事件、取消订阅消息等【收尾工作】。
关于销毁 Vue 实例:
1. 销毁后借助 Vue 开发者工具看不到任何消息。
2. 销毁后自定义事件会失效,但原生 DOM 事件依然有效。
3. 一般不会在 beforeDestory 操作数据,因为即使操作数据,也不会再触发更新流程了。

## 24.Vue 中使用组件
步骤:
1. 定义组件(创建组件)
2. 注册组件
3. 使用组件(写组件标签)
一. 如何定义一个组件?
使用`Vue.extend(options)`创建,其中 `options` 和 `new Vue(options)`时传入的那个 options 几乎一样,但也有点区别;
区别如下:
1. el 不要写,为什么?———— 最终所有的组件都要经过一个 vm 的管理,由 vm 中的 el 决定服务哪个容器。
2. data 必须写成函数,为什么? ———— 避免组件被复用时,数据存在引用关系。
备注:使用 template 可以配置组件结构。
二. 如何注册组件?
1. 局部注册:靠`new Vue`的时候传入`components`选项
2. 全局注册:靠`Vue.component('组件名',组件)`
三. 编写组件标签:
``
```js
//创建hello组件
const hello = Vue.extend({
data() {
return {
msg: "hello",
};
},
template: `
{{msg}}
`,
});
//注册全局组件
Vue.component("hello", hello);
```
## 25.组件的几个注意点
1. 关于组件名:
一个单词组成:
第一种写法(首字母小写):school
第二种写法(首字母大写):School
多个字母组成:
第一种写法(kebab-case命名):my-school
第二种写法(CamelCase命名):MySchool(需要Vue脚手架支持)
备注:
(1). 组件名尽可能回避HTML中已有的元素名称,例如:h2、H2都不行。
(2). 可以使用name配置项指定组件在开发者工具中呈现的名字。
2. 关于自检标签:
第一种写法:``
第二种写法:``
备注:不使用脚手架时,会导致后续组件不能渲染。
3. 一个简写方式:
`const school = Vue.extend(options) 可简写为:const school = options`
## 26. VueComponent
1. school 组件本质是一个名为 VueComponent 的构造函数,且不是程序员定义的,是 Vue.extend 生成的。
2. 我们只需要写``或``,Vue 解析时会帮我们创建 school 组件的实例对象,即 Vue 帮我们执行的:`new VueComponent(options)`。
3. 特别注意:每次调用 Vue.extend,返回的都是一个全新的 VueComponent!!!
4. 关于 this 指向:
(1). 组件配置中:
data函数、methods中的函数、watch中的函数、computed中的函数 它们的this均是【Vuecomponent实例对象】。
(2). new Vue(options)配置中:
data函数、methods中的函数、watch中的函数、computed中的函数 它们的this均是【Vue实例对象】。
5. VueComponent 的实例对象,以后简称 vc(也可称之为:组件实例对象)。
Vue 的实例对象,以后简称 vm。
## 27.一个重要的内置关系
1. 一个重要的内置关系:`VueComponent.prototype.__proto__ === Vue.prototype`
2. 为什么要有这个关系:让组件实例对象(vc)可以访问到 Vue 原型上的属性、方法。
```js
Vue.prototype.x = 99;
const school = Vue.extend({
methods: {
showX() {
// this:VueComponent的实例对象(vc)
console.log(this.x); // 99
console.log(this.__proto__.__proto__ === vm.__proto__); // true
},
},
});
// school:VueComponent
console.log(school.prototype.__proto__ === vm.__proto__); // true
```
