1. 程式人生 > 其它 >vue學習筆記(二)--來源尚矽谷

vue學習筆記(二)--來源尚矽谷

六、資料代理

6.1、回顧Object.defineproperty方法

Object.defineProperty()語法說明
Object.defineProperty()的作用就是直接在一個物件上定義一個新屬性,或者修改一個已經存在的屬性
obj 需要定義屬性的當前物件
prop 當前需要定義的屬性名
desc 屬性描述符
一般通過為物件的屬性賦值的情況下,物件的屬性可以修改也可以刪除,但是通過Object.defineProperty()定義屬性,通過描述符的設定可以進行更精準的控制物件屬性。
屬性的特性以及內部屬性
javacript 有三種類型的屬性
命名資料屬性:擁有一個確定的值的屬性。這也是最常見的屬性
命名訪問器屬性:通過getter

setter進行讀取和賦值的屬性
內部屬性:由JavaScript引擎內部使用的屬性,不能通過JavaScript程式碼直接訪問到,不過可以通過一些方法間接的讀取和設定。
比如,每個物件都有一個內部屬性[[Prototype]],你不能直接訪問這個屬性,但可以通過Object.getPrototypeOf()方法間接的讀取到它的值。
雖然內部屬性通常用一個雙呂括號包圍的名稱來表示,但實際上這並不是它們的名字,它們是一種抽象操作,是不可見的,根本沒有上面兩種屬性有的那種字串型別的屬性

點選檢視程式碼
<body>
		<script type="text/javascript" >
			let number = 18
			let person = {
				name:'張三',
				sex:'男',
			}

			Object.defineProperty(person,'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))

			console.log(person)
		</script>
</body>

6.2、何為資料代理

資料代理:通過一個物件代理對另一個物件中屬性的操作(讀/寫)

點選檢視程式碼
<body>
		
		<script type="text/javascript" >
			let obj = {x:100}
			let obj2 = {y:200}

			Object.defineProperty(obj2,'x',{
				get(){
					return obj.x
				},
				set(value){
					obj.x = value
				}
			})
		</script>
	</body>

6.3、Vue中的資料代理

1.Vue中的資料代理:
通過vm物件來代理data物件中屬性的操作(讀/寫)
2.Vue中資料代理的好處:
更加方便的操作data中的資料
3.基本原理:
通過Object.defineProperty()把data物件中所有屬性新增到vm上。
為每一個新增到vm上的屬性,都指定一個getter/setter。
在getter/setter內部去操作(讀/寫)data中對應的屬性。

點選檢視程式碼
<body>
		<!-- 準備好一個容器-->
		<div id="root">
			<h2>學校名稱:{{name}}</h2>
			<h2>學校地址:{{address}}</h2>
		</div>
</body>

<script type="text/javascript">
		Vue.config.productionTip = false //阻止 vue 在啟動時生成生產提示。
		
		const vm = new Vue({
			el:'#root',
			data:{
				name:'尚矽谷',
				address:'巨集福科技園'
			}
		})
</script>

七、時間處理

7.1、事件的基本使用

事件的基本使用:
1.使用v-on:xxx 或 @xxx 繫結事件,其中xxx是事件名;
2.事件的回撥需要配置在methods物件中,最終會在vm上;
3.methods中配置的函式,不要用箭頭函式!否則this就不是vm了;
4.methods中配置的函式,都是被Vue所管理的函式,this的指向是vm 或 元件例項物件;
5.@click="demo" 和 @click="demo($event)" 效果一致,但後者可以傳參;

點選檢視程式碼
<body>
		<!-- 準備好一個容器-->
		<div id="root">
			<h2>歡迎來到{{name}}學習</h2>
			<!-- <button v-on:click="showInfo">點我提示資訊</button> -->
			<button @click="showInfo1">點我提示資訊1(不傳參)</button>
			<button @click="showInfo2($event,66)">點我提示資訊2(傳參)</button>
		</div>
</body>

<script type="text/javascript">
		Vue.config.productionTip = false //阻止 vue 在啟動時生成生產提示。

		const vm = new Vue({
			el:'#root',
			data:{
				name:'尚矽谷',
			},
			methods:{
				showInfo1(event){
					// console.log(event.target.innerText)
					// console.log(this) //此處的this是vm
					alert('同學你好!')
				},
				showInfo2(event,number){
					console.log(event,number)
					// console.log(event.target.innerText)
					// console.log(this) //此處的this是vm
					alert('同學你好!!')
				}
			}
		})
</script>

7.2、事件修飾符

Vue中的事件修飾符:
1.prevent:阻止預設事件(常用);
2.stop:阻止事件冒泡(常用);
3.once:事件只觸發一次(常用);
4.capture:使用事件的捕獲模式;
5.self:只有event.target是當前操作的元素時才觸發事件;
6.passive:事件的預設行為立即執行,無需等待事件回撥執行完畢;

點選檢視程式碼
<!DOCTYPE html>
<html>
	<head>
		<meta charset="UTF-8" />
		<title>事件修飾符</title>
		<!-- 引入Vue -->
		<script type="text/javascript" src="../js/vue.js"></script>
		<style>
			*{
				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.prevent.stop="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 //阻止 vue 在啟動時生成生產提示。

		new Vue({
			el:'#root',
			data:{
				name:'尚矽谷'
			},
			methods:{
				showInfo(e){
					alert('同學你好!')
					// console.log(e.target)
				},
				showMsg(msg){
					console.log(msg)
				},
				demo(){
					for (let i = 0; i < 100000; i++) {
						console.log('#')
					}
					console.log('累壞了')
				}
			}
		})
	</script>
</html>

7.3、鍵盤事件

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.自定義鍵名 = 鍵碼,可以去定製按鍵別名

點選檢視程式碼
<body>
		<!-- 準備好一個容器-->
		<div id="root">
			<h2>歡迎來到{{name}}學習</h2>
			<input type="text" placeholder="按下回車提示輸入" @keydown.huiche="showInfo">
		</div>
</body>

<script type="text/javascript">
		Vue.config.productionTip = false //阻止 vue 在啟動時生成生產提示。
		Vue.config.keyCodes.huiche = 13 //定義了一個別名按鍵

		new Vue({
			el:'#root',
			data:{
				name:'尚矽谷'
			},
			methods: {
				showInfo(e){
					// console.log(e.key,e.keyCode)
					console.log(e.target.value)
				}
			},
		})
</script>

八、計算屬性

8.1、姓名案例_插值語法實現

點選檢視程式碼
<body>
		<!-- 準備好一個容器-->
		<div id="root">
			姓:<input type="text" v-model="firstName"> <br/><br/>
			名:<input type="text" v-model="lastName"> <br/><br/>
			全名:<span>{{firstName}}-{{lastName}}</span>
		</div>
	</body>

	<script type="text/javascript">
		Vue.config.productionTip = false //阻止 vue 在啟動時生成生產提示。

		new Vue({
			el:'#root',
			data:{
				firstName:'張',
				lastName:'三'
			}
		})
	</script>

8.2、姓名案例_methods實現

點選檢視程式碼
<!DOCTYPE html>
<html>
	<head>
		<meta charset="UTF-8" />
		<title>姓名案例_methods實現</title>
		<!-- 引入Vue -->
		<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>{{fullName()}}</span>
		</div>
	</body>

	<script type="text/javascript">
		Vue.config.productionTip = false //阻止 vue 在啟動時生成生產提示。

		new Vue({
			el:'#root',
			data:{
				firstName:'張',
				lastName:'三'
			},
			methods: {
				fullName(){
					console.log('@---fullName')
					return this.firstName + '-' + this.lastName
				}
			},
		})
	</script>
</html>

8.3、姓名案例_計算屬性實現

計算屬性:
1.定義:要用的屬性不存在,要通過已有屬性計算得來。
2.原理:底層藉助了Objcet.defineproperty方法提供的getter和setter。
3.get函式什麼時候執行?
(1).初次讀取時會執行一次。
(2).當依賴的資料發生改變時會被再次呼叫。
4.優勢:與methods實現相比,內部有快取機制(複用),效率更高,除錯方便。
5.備註:
1.計算屬性最終會出現在vm上,直接讀取使用即可。
2.如果計算屬性要被修改,那必須寫set函式去響應修改,且set中要引起計算時依賴的資料發生改變。

點選檢視程式碼
	<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>{{fullName}}</span> <br/><br/>
			<!-- 全名:<span>{{fullName}}</span> <br/><br/>
			全名:<span>{{fullName}}</span> <br/><br/>
			全名:<span>{{fullName}}</span> -->
		</div>
	</body>

	<script type="text/javascript">
		Vue.config.productionTip = false //阻止 vue 在啟動時生成生產提示。

		const vm = new Vue({
			el:'#root',
			data:{
				firstName:'張',
				lastName:'三',
				x:'你好'
			},
			methods: {
				demo(){
					
				}
			},
			computed:{
				fullName:{
					//get有什麼作用?當有人讀取fullName時,get就會被呼叫,且返回值就作為fullName的值
					//get什麼時候呼叫?1.初次讀取fullName時。2.所依賴的資料發生變化時。
					get(){
						console.log('get被呼叫了')
						// console.log(this) //此處的this是vm
						return this.firstName + '-' + this.lastName
					},
					//set什麼時候呼叫? 當fullName被修改時。
					set(value){
						console.log('set',value)
						const arr = value.split('-')
						this.firstName = arr[0]
						this.lastName = arr[1]
					}
				}
			}
		})
	</script>

8.4、姓名案例_計算屬性實現

點選檢視程式碼
	<body>
		<!-- 準備好一個容器-->
		<div id="root">
			姓:<input type="text" v-model="firstName"> <br/><br/>
			名:<input type="text" v-model="lastName"> <br/><br/>
			全名:<span>{{fullName}}</span> <br/><br/>
		</div>
	</body>

	<script type="text/javascript">
		Vue.config.productionTip = false //阻止 vue 在啟動時生成生產提示。

		const vm = new Vue({
			el:'#root',
			data:{
				firstName:'張',
				lastName:'三',
			},
			computed:{
				//完整寫法
				/* fullName:{
					get(){
						console.log('get被呼叫了')
						return this.firstName + '-' + this.lastName
					},
					set(value){
						console.log('set',value)
						const arr = value.split('-')
						this.firstName = arr[0]
						this.lastName = arr[1]
					}
				} */
				//簡寫
				fullName(){
					console.log('get被呼叫了')
					return this.firstName + '-' + this.lastName
				}
			}
		})
	</script>