vue學習筆記(二)--來源尚矽谷
六、資料代理
6.1、回顧Object.defineproperty方法
Object.defineProperty()語法說明
Object.defineProperty()的作用就是直接在一個物件上定義一個新屬性,或者修改一個已經存在的屬性
obj 需要定義屬性的當前物件
prop 當前需要定義的屬性名
desc 屬性描述符
一般通過為物件的屬性賦值的情況下,物件的屬性可以修改也可以刪除,但是通過Object.defineProperty()定義屬性,通過描述符的設定可以進行更精準的控制物件屬性。
屬性的特性以及內部屬性
javacript 有三種類型的屬性
命名資料屬性:擁有一個確定的值的屬性。這也是最常見的屬性
命名訪問器屬性:通過getter
內部屬性:由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>