Vue - 尚硅谷

The fate of destruction is also the joy of rebirth.

Your effort to remain what you are is what limits you

Vue:一套用于构建用户界面渐进式JS框架
渐进式:如果你已经有一个现成的服务端应用,也就是非单页应用,可以将Vue.js作为该应用的一部分嵌入其中,带来更丰富的交互体验
Vue.js 允许你将一个网页分割成可复用的组件,每个组件都有自己的HTML、CSS、JS来渲染网页中一个对应的位置

如果你构造的是简单应用:只需一个轻量小巧的核心库
如果你构造的是复杂应用:可以引入各式各样的Vue插件

image-20210910170822286

Vue特点
1.采用组件化模式,提高代码复用率、且让代码更好维护
2.一个.vue文件就是一个组件,包含html,css,javascript
3.声明式编码,让编码人员无需直接操作DOM,提高开发效率
4.使用虚拟DOM + 优秀的 Diff 算法,尽量复用 DOM 节点

image-20210904231531963

image-20210904232757546 image-20210904233032566

4.学习Vue之前要掌握哪些JS基础知识?
ES6语法规范
ES6模块化
包管理器
原型、原型链
数组常用方法
axios
promise
......

学习 Vue 用开发版本,Vue 用 script 引入

Vue.config是一个对象,包含Vue的全局配置。可以在启动应用之前修改下列property

阻止Vue在启动之时生成生产提示:

Vue.config.productionTip = false
<!DOCTYPE html>
<html>
	<head>
		<meta charset="utf-8" />
		<title></title>
		<script src="js/vue.js" type="text/javascript" charset="utf-8"></script>
	</head>
	<body>
		<!-- 
		 
		 -->
		
		
		<!-- 准备好一个容器 -->
		<div id="root">
			<h1>Hello,{{name}},{{address}}</h1>
		</div>
		
		<script type="text/javascript">
			Vue.config.productionTip = false;			// 阻止 Vue 在启动时生成生产提示
			
			//创建 Vue 实例
			const new Vue({
				el:'#root',				// el用于指定当前 Vue 实例为哪个容器服务,值通常为css选择器字符串
				data:{						// data中用于存储数据,数据供 el 所指定的容器去使用,值我们暂时先写个对象
					name:'尚硅谷',
					address:'北京昌平'	
				}
			})
		</script>
	</body>
</html>

初识Vue
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(){}

模板语法

<!DOCTYPE html>
<html>
	<head>
		<meta charset="utf-8" />
		<title></title>
		<script src="js/vue.js" type="text/javascript" charset="utf-8"></script>
	</head>
	<body>
	<h1>
		<div id="root">
			<h1>插值语法</h1>
			<h3>你好,{{name}}</h3>
			<br>
			<h1>指令语法</h1>
			<a :href="school.url">点我去{{school.name}}学习1</a>
		</div>
	</h1>
		<script type="text/javascript">
			Vue.config.productionTip = false;
			
			new Vue({
				el:'#root',
				data:{
					name:'Jack',
          school:{
            name:'尚硅谷',
            url:'http://www.atguigu.com',
          }
				}
			})
		</script>
	</body>
</html>

Vue模板语法有2大类:

1.插值语法:
功能:用于解析标签体内容
写法:{{xxx}},xxx 是 js 表达式,且可以直接读取到data中的所有属性

2.指令语法:
功能:用于解析标签(包括:标签属性、标签体内容、绑定事件......)
举例:v-bind:href="xxx" 或 简写为 :href="xxx",xxx 同样要写js表达式,且可以直接读取到data中的所有属性
备注:Vue中有很多的指令,且形式都是:v-????,此处我们只是拿v-bind举个例子

<!DOCTYPE html>
<html>
	<head>
		<meta charset="utf-8" />
		<title></title>
		<script src="./js/vue.js" type="text/javascript" charset="utf-8"></script>
	</head>
	<body>
		<div id="root">
			<!-- 普通写法 -->
			<!-- 单向数据绑定:<input type="text" v-bind:value="name"><br>
			双向数据绑定:<input type="text" v-model:value="name"> -->
			
			<!-- 简写 -->
			单向数据绑定:<input type="text" :value="name"><br>
			双向数据绑定:<input type="text" v-model="name">
			<!-- 如下代码是错误的,因为v-model只能应用在表单类元素(输入类元素)上 -->
			<!-- <h2 v-model:x="name">你好</h2> -->
		</div>
		<script type="text/javascript">
			Vue.config.productionTip = false;	//阻止 Vue 在启动时生成生产提示
			
			new Vue({
				el:'#root',
				data:{
					name:"尚硅谷"
				}
			})
		</script>
	</body>
</html>

Vue 有2中数据绑定的方式:
1.单向绑定(v-bind):数据只能从data流向页面
2.双向绑定(v-model):数据不仅能从data流向页面,还可以从页面流向data

备注:
1.双向绑定一般都应用在表单类元素上(如:input、select等)
2.v-model:value 可以简写为 v-model 因为 v-model 默认收集的就是 value 值
3.v-bind: 简写 :
4.v-on: 简写 @

<!DOCTYPE html>
<html lang="zh">
<head>
	<meta charset="UTF-8">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<meta http-equiv="X-UA-Compatible" content="ie=edge">
	<title></title>
	<script type="text/javascript" src="./js/vue.js">
		
	</script>
</head>
<body>
	<div id="root">
		<h1>你好,{{name}}</h1>
	</div>
	
	<script type="text/javascript">
		Vue.config.productionTip = false;
		
		/*el的两种写法
		 const v = new Vue({
			 el:'#root';			//第一种写法
			 data:{
				 name:'尚硅谷'
			 }
		 })
		 v.$mount('#root')		//第二种写法  */
		
    
		//data 的两种写法
		new Vue({
			el:'#root',
			//data的第一种写法:对象式
			/*
      data:{
			 	name:'尚硅谷'
			 }
			 */
			
			// data的第二种写法:函数式
      // date属性的值,是一个函数,这个函数是Vue调用的,
			data(){
				console.log('@@@',this)		//此处的this是Vue实例对象				箭头函数没有自己的this,往外找会找到window
				return{
					name:'尚硅谷'
				}
			}
		})*/
	</script>
</body>
</html>

总结

data 与 el 的2种写法

1.el有2中写法
(1) new Vue 的时候配置el属性
(2) 先创建Vue实例,随后再通过v.$mount('#root') 指定 el 的值

2.data有2种写法
(1)对象式
(2)函数式

如何选择:目前哪种写法都可以,以后学习到组件时,data必须使用函数式,否则会报错

3.一个重要的原则:由Vue管理的函数,一定不要写箭头函数,一旦写了箭头函数,this 就不再是Vue实例了

MVVM 模型

  1. M:模型(Model):对应 data 中的数据
  2. V:视图(View):模板
  3. VM:视图模型(ViewModel):Vue实例对象(简称vm) --- 以后拿变量接收Vue实例,需要用变量 vm
image-20210909103728712

把一堆乱七八糟的数据和一堆DOM结构,它在中间做了一个连接,它是中间的桥梁和纽带

前端的主流框架设计都是这个思想:你把数据放在我要求放好的位置,然后 你写出这种模板代码,模板里面具体怎么往里面插入值,需要学习 具体哪种框架的语法了

image-20210909104302433

<!DOCTYPE html>
<html lang="zh">
<head>
	<meta charset="UTF-8">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<meta http-equiv="X-UA-Compatible" content="ie=edge">
	<title></title>
	<script type="text/javascript" src="./js/vue.js"></script>
</head>
<body>
	<div id="root">
		<h1>学校名称:{{name}}</h1>
		<h1>学校地址:{{address}}</h1>
		<!-- <h1>{{$emit}}</h1> -->
	</div>
	
	<script type="text/javascript">
		Vue.config.productionTip = false
		const vm = new Vue({
			el:'#root',
			data:{
				name:'尚硅谷',
				address:'北京'
			}
		})
		console.log(vm);
	</script>
</body>
</html>

总结:

MVVM 模型

  1. M:模型,data中的数据
  2. V:视图,模板代码
  3. VM:视图模型,Vue实例

data 中的所有属性,最后都出现在了vm身上
vm身上所有的属性,及 Vue 原型上所有属性,在Vue模板中都可以直接使用

数据代理

回顾Object.defineproperty方法:给一个对象定义属性用的

image-20210909112314676

Object.keys() 要传入一个对象作为参数,这个方法可以把传入对象的所有属性的属性名提取出来,变成一个数组

<!DOCTYPE html>
<html lang="zh">
<head>
	<meta charset="UTF-8">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<meta http-equiv="X-UA-Compatible" content="ie=edge">
	<title></title>
	<script type="text/javascript" src="./js/vue.js"></script>
</head>
<body>
	
	<script type="text/javascript">
		Vue.config.productionTip = false
		
		let number = 18
		let person = {
			name:'张三',
			sex:'男',
			// age:18
		}
		
		Object.defineProperty(person,'age',{
			// person确实是age对象,确实有age属性,值现用,我现取
			// value:18,
			// enumerable:true,	//控制 属性是否可以枚举,默认值是false
			// writable:true,		//控制 属性是否可以被修改,默认值是false
			// configurable: true		//控制 属性是否可以被删除,默认值是false
			
			// 当有人读取person的age属性时,get函数(getter)就会被调用,且返回值就是age的值
			get() {
				console.log('有人读取age属性了')
				return number
			},
			
			// 当有人修改person的age属性时,set函数(setter)就会被调用,且会收到修改的具体值
			set(value) {
				console.log('有人修改了age属性,且值是',value)
				number = value
			}
		})
		
		// console.log(Object.keys(person));
		// for in:既能遍历数组,也能遍历对象
		
		// console.log(person);
	</script>
</body>
</html>

数据代理:通过一个对象代理对另一个对象中属性的操作(读/写)

<script>
	let obj = {x:100}
  let obj2 = {y:200}
  
  Object.defineProperty(obj2,'x',{
    get() {
      return obj.x
    },
    set(value) {
      obj.x = value
    }
  })
</script>
image-20210909191809304

vm._data.name = vm.name

image-20210909194043103
<!DOCTYPE html>
<html lang="zh">
<head>
	<meta charset="UTF-8">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<meta http-equiv="X-UA-Compatible" content="ie=edge">
	<title>Vue中的数据代理</title>
	<script type="text/javascript" src="./js/vue.js"></script>
</head>
<body>
	<div id="root">
		<h2>学校名称:{{name}}</h2>
		<h2>学校地址:{{address}}</h2>
	</div>
</body>

<script type="text/javascript">
		Vue.config.productionTip = false
		const vm = new Vue({
			el:'#root',
			data:{
				name:'尚硅谷',
				address:'宏福科技园'
			}
		})
	</script>
</html>

1.Vue 中的数据代理:

通过 vm 对象来代理 data 对象中属性的操作(读/写)

2.Vue 中数据代理的好处:

更加方便地操作 data 中的数据

3.基本原理:

通过 Object.defineProperty() 把 data 对象中所有属性添加到 vm 上
为每一个添加到 vm 上的属性,都指定一个 getter / setter 
在 getter / setter 内部去操作(读/写) data 中对应的属性

模板只能使用Vue实例所提供的东西

所有被 Vue 管理的函数最好都写成普通函数

<!DOCTYPE html>
<html lang="zh">
<head>
	<meta charset="UTF-8">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<meta http-equiv="X-UA-Compatible" content="ie=edge">
	<title></title>
	<script type="text/javascript" src="./js/vue.js"></script>
</head>
<body>
	<div id="root">
		<h2>欢迎来到{{name}}学习</h2>
		<button @click="showInfo1">点我提示信息1(不传参)</button>
		<button @click="showInfo2($event,66)">点我提示信息2(传参)</button>
	</div>
</body>
	
	<script type="text/javascript">
		Vue.config.productionTip = false;
		new Vue ({
			el:"#root",
			data:{
				name:'尚硅谷',
			},
			methods:{
				showInfo1(event){
					alert('同学你好!')
				},
				showInfo2(event){
					console.log(event,number)
					//console.log(event.target.innerText)
					//console.log(this)	//此处的 this 是 vm
					alert('同学你好!!')
				}
			}
		})
	</script>
</html>

总结:

事件的基本使用:

1.使用 v-on:xxx 或@xxx 绑定事件,其中 xxx 是事件名
2.事件的毁掉需要配置在 methods 对象中,最终会在 vm 上
3.methods 中配置的函数,不要用箭头函数,否则 this 就不是 vm 了
4.methods 中配置的函数,都是被 Vue 所管理的函数,this 的指向是 vm 或 组件实例对象;
5.@click="demo" 和 @click = "demo($event)" 效果一致,但后者可以传参;

<!DOCTYPE html>
<html lang="zh">
<head>
	<meta charset="UTF-8">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<meta http-equiv="X-UA-Compatible" content="ie=edge">
	<title></title>
	<script type="text/javascript" src="./js/vue.js"></script>
</head>
<body>
	<div id="root">
		<h2>欢迎来到{{name}}学习</h2>
		<a href="http://www.atguigu.com" @click.prevent="showInfo">点我提示信息</a>
	</div>
</body>
<script type="text/javascript">
	Vue.config.productionTip = false
	
	new Vue({
		el:'#root',
		data:{
			name:'尚硅谷'
		},
		methods:{
			showInfo(e){
				alert('同学你好!')
			}
		}
	})
</script>
</html>

Vue事件修饰符
1.prevent:阻止默认事件(常用)
2.stop:阻止事件冒泡(常用)
3.once:事件只触发一次(常用)
4.capture:使用事件的捕获模式
5.self:只有 event.target 是 当前操作的元素 时,才触发事件
6.passive:事件的默认行为立即执行,无需等待事件回调执行完毕

事件捕获阶段从外到内

事件冒泡阶段由内向外

@scroll:滚动条的滚动

@wheel:滚动轮的滚动

<!DOCTYPE html>
<html lang="zh">
<head>
	<meta charset="UTF-8">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<meta http-equiv="X-UA-Compatible" content="ie=edge">
	<title></title>
	<script type="text/javascript" src="./js/vue.js"></script>
	<style type="text/css">
	*{
		margin-top: 20px;
	}
		.demo1{
			height: 50px;
			background-color: skyblue;
		}
		
		.box1{
			padding:5px;
			background-color: skyblue;
		}
		.box2{
			padding:5px;
			background-color: orange;
		}
		
		.list{
			width: 200px;
			height: 200px;
			background-color: peru;
			overflow: auto;
		}
		
		li{
			height: 100px;
			
		}
	</style>
</head>

<body>
	<div id="root">
		<h2>欢迎来到{{name}}学习</h2>
		<!-- 阻止默认事件(常用) -->
		<a href="http://www.atguigu.com" @click.prevent="showInfo">点我提示信息</a>
		<!-- 阻止事件冒泡(常用) -->
		<div class="demo1" @click="showInfo">
			<button @click.stop="showInfo">点我提示信息</button>
			
			<!-- 修饰符可以连续写 -->
			<!-- <a href="http://www.atguigu.com" @click.stop.prevent="showInfo">点我提示信息</a> -->
      
		</div>
		<!-- 事件只触发一次(常用) -->
		<button @click.once="showInfo">点我提示信息</button>
		<!-- 使用事件的捕获模式 -->
		<div class="box1" @click.capture="showMsg(1)">
			div1
			<div class="box2" @click="showMsg(2)">
				div2
			</div>
		</div>
		
		<!-- 只有 event.target 是 当前操作的元素 时,才触发事件 -->
		<div class="demo1" @click.self="showInfo">
			<button @click="showInfo">点我提示信息</button>
		</div>
		
		<!-- 事件的默认行为立即执行,无需等待事件回调执行完毕 -->
		<ul @wheel.passive="demo" class="list">
			<li>1</li>
			<li>2</li>
			<li>3</li>
			<li>4</li>
		</ul>
	</div>
</body>
<script type="text/javascript">
	Vue.config.productionTip = false
	
	new Vue({
		el:'#root',
		data:{
			name:'尚硅谷'
		},
		methods:{
			showInfo(e){
				alert('同学你好!')
			},
			showMsg(msg){
				console.log(msg)
			},
				demo(){
					for (let i = 0; i < 100000; i++) {
						console.log("#");
					}
					console.log("累坏了");
				}
		}
	})
</script>
</html>
  1. Vue 中常用的按键别名:
    回车 enter
    删除 delete(捕获“删除” 和 “退格”键)
    退出 esc
    空格 space
    换行 tab(必须配合 keydown 使用)
    上 up
    下 down
    左 left
    右 right

  2. Vue 未提供别名的案件,可以使用按键原始的 key 值去绑定,但注意要转为 kebab-case(短横线命名)

  3. 系统修饰键(用法特殊):ctrl、alt、shift、meta
    (1).配合 keyup 使用:按下修饰键的同时,再按下其他键,随后释放其他键,事件才被触发
    (2).配合 keydown 使用:正常触发事件

  4. 也可以使用 keyCode 去指定具体的案件(不推荐)

  5. Vue.config.keyCodes.自定义键名 = 键码,可以去定制按键别名

键盘上任何一个按键都有自己的名字和编码( Enter 13)

<!DOCTYPE html>
<html lang="zh">
<head>
	<meta charset="UTF-8">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<meta http-equiv="X-UA-Compatible" content="ie=edge">
	<title></title>
	<script type="text/javascript" src="./js/vue.js"></script>
</head>
<body>
	<div id="root">
		<h2>欢迎来到{{name}}学习</h2>
		<input type="text" placeholder="按下回车提示输入" @keyup.ctrl.y="showInfo">
	</div>
</body>
	
	<script type="text/javascript">
		Vue.config.productionTip = false;
		// Vue.config.keyCodes.huiche = 13;		//定义了一个别名按键
		new Vue ({
			el:"#root",
			data:{
				name:'尚硅谷',
			},
			methods:{
				showInfo(e) {
					
					console.log(e.target.value)
				}
				
			}
		})
	</script>
</html>

计算属性 - computed

页面上的输入要影响数据里的变化,需要用 v-model

插值语法实现姓名案例

<!DOCTYPE html>
<html lang="zh">
<head>
	<meta charset="UTF-8">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<meta http-equiv="X-UA-Compatible" content="ie=edge">
	<title>插值语法实现</title>
	<script type="text/javascript" src="./js/vue.js"></script>
</head>
<body>
	<div id="root">
		姓:<input type="text" v-model="firstName"> <br><br>	
		名:<input type="text" v-model="lastName"> <br><br>
		全名:<span id="">{{firstName}}-{{lastName}}</span>
	</div>
</body>
	<script type="text/javascript">
		Vue.config.productionTip = false;
		
		new Vue ({
			el:"#root",
			data:{
				firstName:'张',
				lastName:'三',
				
			},
			methods:{
				
				}
				
			})
	</script>
</html>

只要data中的数据发生改变,Vue一定会重新解析模板

<!DOCTYPE html>
<html lang="zh">
<head>
	<meta charset="UTF-8">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<meta http-equiv="X-UA-Compatible" content="ie=edge">
	<title>methods实现</title>
	<script type="text/javascript" src="./js/vue.js"></script>
</head>
<body>
	<div id="root">
		姓:<input type="text" v-model="firstName"> <br><br>	
		名:<input type="text" v-model="lastName"> <br><br>
		全名:<span id="">{{fullName()}}</span>
	</div>
</body>
	<script type="text/javascript">
			
		Vue.config.productionTip = false
		
		new Vue ({
			el:"#root",
			data:{
				firstName:'张',
				lastName:'三',
			},
				methods:{
					fullName() {
						return this.firstName + '-' + this.lastName
					}
			},
		})
	</script>
</html>

Vue 认为 data 里面写的东西都是属性

计算属性 就是拿着你已经写好的属性 去加工、计算,然后生成一个全新的属性

<!DOCTYPE html>
<html lang="zh">
<head>
	<meta charset="UTF-8">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<meta http-equiv="X-UA-Compatible" content="ie=edge">
	<title>计算属性实现</title>
	<script type="text/javascript" src="./js/vue.js"></script>
</head>
<body>
	<div id="root">
		姓:<input type="text" v-model="firstName"> <br><br>	
		名:<input type="text" v-model="lastName"> <br><br>
		测试:<input type="text" v-model="x"> <br><br>
		全名:<span id="">{{fullName}}</span>
	</div>
</body>
	<script type="text/javascript">
			
		Vue.config.productionTip = false
		
		const vm = new Vue ({
			el:"#root",
			data:{
				firstName:'张',
				lastName:'三',
				x:'你好'
			},
			computed:{
				fullName:{
					// get有什么用:当有人读取fullName时,get就会被调用,且返回值就作为fullName的值
					// get什么时候调用:1.初次读取 fullName 时	2.所依赖的数据发生变化时
					get(){
						console.log('此时get被调用');
						return this.firstName + '-' + this.lastName;
					},
					// set什么时候调用:当 fullName 被修改时
					set(value) {
						const arr = value.split('-')
						this.firstName = arr[0]
						this.lastName = arr[1]
					}
				}
			}
		})
	</script>
</html>

计算属性:

1.定义:要用的属性不存在,要通过已有属性计算得来

2.原理:底层借助了Object.defineproperty 方法提供的 getter 和 setter

3.get函数什么时候执行?

​ 1.初次读取时会执行1次
​ 2.当依赖的数据法身改变时会被再次调用

4.优势:与methods实现相比,内部有缓存机制(复用),效率更高,调试方便

5.备注:

​ 1.计算属性最终会出现在vm上,直接读取使用即可
​ 2.如果计算属性要被修改,那必须写set函数去相应修改,且 set 中要引起计算时依赖的数据发生变化

计算属性如果只读不改的话,可以简写

<!DOCTYPE html>
<html lang="zh">
<head>
	<meta charset="UTF-8">
	<meta name="viewport" content="width=device-width, initial-scale=1.0">
	<meta http-equiv="X-UA-Compatible" content="ie=edge">
	<title>计算属性简写实现</title>
	<script type="text/javascript" src="./js/vue.js"></script>
</head>
<body>
	<div id="root">
		姓:<input type="text" v-model="firstName"> <br><br>	
		名:<input type="text" v-model="lastName"> <br><br>
		全名:<span id="">{{fullName}}</span><br><br>
	</div>
</body>

	<script type="text/javascript">
		Vue.config.productionTip = false
		
		const vm = new Vue ({
			el:"#root",
			data:{
				firstName:'张',
				lastName:'三',
			},
			computed:{
				
				// 完整写法
				// fullName:{
				// 	get(){
				// 		console.log('此时get被调用');
				// 		return this.firstName + '-' + this.lastName
				// 	}
				//}
					// 简写(只有不考虑修改只考虑读取的时候可以使用
					fullName:function() {
								console.log('此时get被调用');
								return this.firstName + '-' + this.lastName;
					},
				}
		})
	</script>
</html>
posted @ 2021-09-08 20:47  ztw1002  阅读(113)  评论(0编辑  收藏  举报