跳至主要內容

Vue 核心

T4makoVue基础语法Vue大约 24 分钟

Vue 核心

1.1、Vue 对象

想要 Vue 工作必须要创建 Vue 实例,并传入一个 配置对象

el :绑定容器
容器Vue实例 关系:一对一
Vue实例组件 配合使用
容器中的代码符合 html 规范,并且加入了特殊的 Vue 语法 (Vue 模板

<!-- 创建一个 root 容器 -->
<div id="root">
    <h1>GoodBye {{name}},{{age}},{{Date.now()}},</h1> 
</div>


<script type="text/javascript">

    // 创建 Vue 实例
    new Vue({

        // el 用于指定当前 Vue 实例为哪个容器服务
        // 值 通常 为 css 选择器
        el: '#root', // id 为 root 的元素

        // data 中用于存储数据,数据供 el 所指定的容器使用
        // 值 可以写成 对象,函数
        data: {
            name: 't4mako',
            age: 18
        }
    })
</script>

1.2、el 与 data 的两种写法

el 的 两种写法:
① new Vue() 时配置
② 先创建实例,通过 vm.$mount('#root') 指定

data 的 两种写法:
对象式
函数式

注:由 Vue 管理的函数,不要写箭头函数,this 就不再是Vue实例了( this 指向问题 window)

<div id="root">
    <h2>{{name}}</h2>
</div>

<script>
    const vm = new Vue({
        // 写法一
        //el: '#root', 
        // data: {
        //     name: 'hello'
        // }

        // 写法二
        // 此处:functioin可省略:
        // data(){return{ name: 'hello'}}
        data:function(){
            return{
                name: 'hello'
            }
        }
    })
    vm.$mount('#root') // 写法二
</script>

1.3、Vue 模板语法

插值语法指令语法

差值语法
解析 标签体内容
语法:
双括号内只能写 Js表达式

指令语法:
解析 标签属性解析标签体内容绑定事件
如:v-bind:href = 'xxxx (v-bind: 可简写成 : )

<div id="root">
    <h1>插值语法</h1> 
    <h3>你好,{{name}}</h3>

    <h1>指令语法</h1>
    <a v-bind:href="url">bilibili 1</a>
    <a :href="url">bilibili 2</a> <!-- v-bind 可以简写成 : -->
</div>


<script type="text/javascript">
    // 创建 Vue 实例
    new Vue({
        el: '#root',             
        data: {
            name: 'Tom',
            url: 'http://www.bilibili.com'
        }
    })
</script>

1.4、v-bind/model 数据绑定

1.4.1、v-bind 单项数据绑定

v-bind:单项数据绑定

image-20230509152524615

1.4.2、v-model 双向数据绑定

v-model:双向数据绑定

v-model 只能应用在 表单类元素 上(输入类元素 input select)
v-model:value 可以简写成 v-model,因为 v-model 默认收集的就是 value 值

<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"><br>

</div>

<script>
    new Vue({
        el: '#root',
        data: {
            name: 'hello'
        }
    })
</script>

1.5、MVVM 理解

Vue 参考了 MVVM 模型

M:模型 model (对应 data 中的数据 )
V:视图 View (模板 )
VM:视图模型 ViewModel (Vue 实例对象)

1.6、数据代理

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

1.6.1、回顾 defineProperty()

回顾 ES6语法:defineProperty()open in new window
Object.defineProperty() 方法会直接在一个对象上定义一个新属性,或者修改一个对象的现有属性,并返回此对象。

<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>

通过 Object.defineProperty() 实现数据代理:

<!-- 数据代理:通过一个对象代理对另一个对象中属性的操作(读/写)-->
<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>

1.6.2、Vue 中的数据代理

data 中的值都会有对应的 getter setter 方法

创建 vm对象 时,对象中 data 的数据最后会在 _data 中(数据劫持)
_data 中的数据会再 添加到 vm对象 中

1.7、v-on/@ 事件处理

1.7.1、事件的基本使用

​ 绑定事件: v-on:xxx 或者 @xxx 绑定事件
​ @xxx = “yyy”,""内可以写一些简单的语句,对应 vm 身上的属性与方法
​ 事件的回调需配置在 methods 对象中,最终会在 vm 上
​ methods 中的函数不要写箭头函数,( this )
​ methods 中的函数被 vue 管理,this 指向 vm 或 组件实例对象}
​ 参数传递:@click="demo(event)"event)" (event 为当前dom)

<div id="a">
    <h2>welcome {{name}}</h2>
    
    <button @click="show">click</button> 
    
    <!-- 普通写法 -->
    <button v-on:click="showInfo">click me1</button>
    <!-- 简写形式 -->
    <button @click="showInfo">click me2</button> 
    <!-- 参数传递 -->
    <button @click="showInfo1($event,66)">click me3</button> 
</div>
<script>
    const vm = new Vue({
        el: '#a',
        data: {
            name: 'T4mako'
        },
        methods: {
            show(){
                alert('11')
            },
            
            showInfo(event){ // 不要写成箭头函数
                console.log(event);
                console.log(event.target.innerText);
                console.log(this); //此处的this是vm
                alert('hello');
            },
            showInfo1(event,number){
                console.log(number);
                console.log(event);
            }
        }
    })
</script>

1.7.2、事件修饰符

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

<div id="root">
    
    <!-- 阻止默认事件(常用) -->
    <a href="http://www.bilibili.com" @click.prevent="showInfo">click me</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)">
        <!-- 点击输出 1 2 -->
        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:'AAA'
        },
        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>

1.7.3、键盘事件

① Vue中 常用按键别名
回车 => enter
删除 => delete (捕获“删除”和“退格”键)
退出 => esc
空格 => space
换行 => tab (特殊,必须配合keydown去使用)
上 => up
下 => down
左 => left
右 => right

② ctrl、alt、shift、meta(Win键):
配合 keyup 使用:按下这些键的同时,再按下其他键,随后释放其他键,事件才被触发。
配合 keydown 使用:正常触发事件。

③ 也可以使用keyCode去指定具体的按键(不推荐)

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

⑤ 修饰符可以 链式 调用
@click.prevent.stop = "show"
@click.ctrl.s = "show"

<div id="root">
    
    <input type="text" placeholder="enter" @keyup.enter="show">
    <button @click.prevent.stop="show" value="123">click me</button>
    <input type="text" @keyup.huiche="show">
</div>

<script>

    Vue.config.keyCodes.huiche = 13 //定义了一个别名按键

    new Vue({
        el: '#root',
        data: {},
        methods: {
            show(e){
                console.log(e.target.value);
            }
        }
    })
</script>

1.8、计算属性computed 与 监视watch

computedwatch 配置项

1.8.1、计算属性

通过 已有属性 计算得来 要用的属性
原理:底层借助了Objcet.defineproperty 方法提供的getter和setter
内部有 缓存机制(复用),效率更高,调试方便
get 函数执行时间:
初次读取时会执行一次
当依赖的数据发生改变时会被再次调用

计算属性 最终会出现在 vm 上,直接读取 使用即可
如果计算属性要被修改,那必须写 set函数 去响应修改 (改 data 中数据)

<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">

    const vm = new Vue({
        el:'#root',
        data:{
            firstName:'张',
            lastName:'三',
        },
        // 计算属性
        computed:{
            fullName:{
                // 初次读取 fullName 时。调用哦 get
                // 所依赖的数据发生变化时 调用 get (否则为缓存)
                // get 返回值就作为 fullName 的值
                // get 中的 this 指向 vm
                get(){
                    console.log('get被调用了')
                    // console.log(this) //此处的this是vm
                    return this.firstName + '-' + this.lastName
                },
                // set 非必须的
                // 当fullName被修改调用set
                // set 中的 this 指向 vm
                set(value){
                    console.log('set',value)
                    const arr = value.split('-')
                    this.firstName = arr[0]
                    this.lastName = arr[1]
                }
            }
        }
    })
</script>

计算属性简写:

没有 set 方法时可以简写

<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:{
            // 简写
            // 函数就当 get() 用
            fullName(){
                console.log('get被调用了')
                return this.firstName + '-' + this.lastName
            }
            //完整写法
            /* 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]
					}
				} */
        }
    })
</script>

1.8.2、监视

1.8.2.1、监视属性

配置项:watch
① 当 被监视的属性变化, 回调函数调用
② 监视的属性必须存在,才能进行监视
③ 监视的两种写法:
(1) new Vue时传入watch 配置
(2) 通过 vm.$watch 监视

<script>
    const vm = new Vue({
        el: "#root",
        data: {
            isHot: true
        },
        computed: {
            info(){
                return this.isHot ? 'hot' : 'cold'
            }
        },
        methods: {
            changeWeather(){
                this.isHot = !this.isHot
            }
        },
        
        // ① 通过 watch 实现监视
        // watch: {
        //     isHot: {
        //         immediate: true, // 初始化时让 handler() 调用
        //         // 当 isHot 发生改变时 调用 handler()
        //         handler(newValue,oldValue){
        //             console.log('isHot changed',newValue,oldValue);
        //         }

        //     }
        // }
    })

    // ② 通过 $watch 监视
    vm.$watch('isHot', {
        immediate: true, // 初始化时让 handler() 调用
        // 当 isHot 发生改变时 调用 handler()
        handler(newValue,oldValue){
            console.log('isHot changed',newValue,oldValue);
        }
    })
</script>

1.8.2.2、深度监视

深度监视:
Vue中的watch默认不监测对象内部值的改变(只监视一层)。
配置 deep:true 可以监测对象内部值改变(多层)。

注:
Vue自身可以监测对象内部值的改变,但 Vue 提供的 watch 默认不可以
使用 watch 时根据数据的具体结构,决定是否采用深度监视。

<div id="root">
    <h3>a 的值:{{numbers.a}}</h3>
    <button @click="numbers.a++">click me let a++</button>
    <h3>b 的值:{{numbers.b}}</h3>
    <button @click="numbers.b++">click me let b++</button>
    <h3>numbers 的值:{{numbers}}</h3>
    <button @click="numbers = {a:666,b:888}">click me change numbers</button>
</div>

<script>
    const vm = new Vue({
        el: "#root",
        data: {
            isHot: true,
            numbers: {
                a: 1,
                b: 2
            }
        },
        computed: {
            info(){
                return this.isHot ? 'hot' : 'cold'
            }
        },
        methods: {
        },
        // 通过 watch 实现监视
        watch: {

            // 深度监视 a (必须加引号)
            // 'numbers.a':{
            //     handler(){
            //         console.log('a changed');
            //     }
            // },

            // deep 深度监视
            // 监视多级结构中所有属性的变化
            numbers: {
                deep: true, // 开启深度监视
                handler(){
                    console.log('numbers changed');
                }
            }
        }
    })
</script>

1.8.2.3、监视属性简写

watch 配置项 简写

watch: {
    // 简写
    // 相当于 handler()
    isHot(newValue,oldValue){
        console.log('changed',newValue,oldValue);
    }
    
    // 正常写法
    /*isHot:{
        // immediate: true, 初始化调用
        // deep:true,
        handler(newValue,oldValue){
                        console.log('changed',newValue,oldValue);
		}
	}*/
}

vm.$watch 简写

//简写
vm.$watch('isHot', function(newValue,oldValue){
    console.log(newValue,oldValue);
})

// 正常写法
vm.$watch('isHot', {
    immediate: true, // 初始化时让 handler() 调用
    // 当 isHot 发生改变时 调用 handler()
    handler(newValue,oldValue){
        console.log('isHot changed',newValue,oldValue);
    }
})

1.8.3、computed 和 watch 区别

computed和watch之间的区别:
1.computed能完成的功能,watch都可以完成。
2.watch能完成的功能,computed不一定能完成,例如:watch可以进行异步操作

两个重要的小原则:
1.被Vue管理的函数,最好写成普通函数,这样this的指向才是 vm 或 组件实例对象。
2.所有 不被Vue所管理的函数(定时器的回调函数、ajax的回调函数等、Promise的回调函数),最好 写成箭头函数,这样this的指向才是vm 或 组件实例对象。

1.9、:class/style 绑定样式

class 样式绑定:
写法 :class="xxx" xxx可以是 字符串、对象、数组 (:代表 v-bind)

style 样式:
:style = "{fontSize: xxx}" 其中xxx是动态值。{ } 内是 js 的表达式
:style = "[a,b]" 其中a、b是样式对象。

<div id="root">
    <!-- 绑定 class 样式 -->
    <!-- 字符串 写法
		适用于:样式类名不确定,需动态指定 -->
    <!-- : 代表 v-bind -->
    <div class="basic" :class="mood" @click="changeMood">{{name}}</div> <br/><br/>

    <!-- 数组 写法
		适用于:要绑定的样式个数不确定、名字也不确定 -->
    <div class="basic" :class="classArr">{{name}}</div> <br/><br/>

    <!-- 对象 写法
		适用于:要绑定的样式个数确定、名字也确定,但要动态决定用不用 -->
    <div class="basic" :class="classObj">{{name}}</div> <br/><br/>


    
    <!-- 绑定 style 样式 -->
    <!-- 对象 写法 -->
    <div class="basic" :style="styleObj">{{name}}</div> <br/><br/>
    
    <!-- 数组 写法 -->
    <div class="basic" :style="styleArr">{{name}}</div>
</div>

<script type="text/javascript">

    const vm = new Vue({
        el:'#root',
        data:{
            name:'t4mako',
            mood:'normal',
            classArr:['atguigu1','atguigu2','atguigu3'],
            classObj:{
                atguigu1:false,
                atguigu2:false,
            },
            styleObj:{
                fontSize: '40px',
                color:'red',
            },
            styleObj2:{
                backgroundColor:'orange'
            },
            styleArr:[
                {
                    fontSize: '40px',
                    color:'blue',
                },
                {
                    backgroundColor:'gray'
                }
            ]
        },
        methods: {
            changeMood(){
                const arr = ['happy','sad','normal']
                const index = Math.floor(Math.random()*3)
                this.mood = arr[index]
            }
        },
    })
</script>

1.10、条件渲染

1、 v-show

​ 写法:v-show = "表达式"
​ 适用于:切换频率较高 的场景。
​ 特点:不展示的DOM元素未被移除,仅仅是使用样式隐藏掉 (display = none)

<h2 v-show="false">欢迎{{name}}</h2>
<h2 v-show="1 === 1">欢迎{{name}}</h2>

2、 v-if

写法:
① v-if = "表达式"
② v-else-if = "表达式"
③ v-else = "表达式"
适用于:切换频率较低的场景。
特点:不展示的DOM元素直接被移除。

注:v-if 可以和 v-else-if、v-else 一起使用,但要求结构 不能被“打断”(处在相同 if else 的元素要紧挨在一起)

<div id="root">
    <h2>当前的n值是:{{n}}</h2>
    <button @click="n++">点我n+1</button>
    <!-- 使用v-show做条件渲染 -->
    <!-- <h2 v-show="false">欢迎来到{{name}}</h2> -->
    <!-- <h2 v-show="1 === 1">欢迎来到{{name}}</h2> -->

    <!-- 使用v-if做条件渲染 -->
    <!-- <h2 v-if="false">欢迎来到{{name}}</h2> -->
    <!-- <h2 v-if="1 === 1">欢迎来到{{name}}</h2> -->

    <!-- v-else和v-else-if -->
    <!-- <div v-if="n === 1">Angular</div>
	<div v-else-if="n === 2">React</div>
	<div v-else-if="n === 3">Vue</div>
	<div v-else>哈哈</div> -->

    <!-- v-if与template的配合使用 -->
    <template v-if="n === 1">
        <h2>你好</h2>
        <h2>尚硅谷</h2>
        <h2>北京</h2>
    </template>

</div>
</body>
<script type="text/javascript">
    Vue.config.productionTip = false

    const vm = new Vue({
        el:'#root',
        data:{
            name:'city',
            n:0
        }
    })
</script>

内容模板(<template>不影响结构,该元素是一种用于保存客户端内容机制,该内容在加载页面时不会呈现,但随后可以 在运行时使用 JavaScript 实例化。

1.11、v-for 列表渲染

1.11.1、基本列表

v-for
用于 展示列表数据
语法:**v-for = "(item, index) in xxx" :key = "yyy" **(in 可以换成 of)
可遍历:数组、对象、字符串(用的很少)、指定次数(用的很少)

<div id="root">
    <!-- 遍历数组 -->
    <h2>遍历数组</h2>
    <ul>
        <li v-for="(p,index) in persons" :key="index">
            {{p.name}}-{{p.age}}
        </li>
    </ul>

    <!-- 遍历对象 -->
    <h2>遍历对象</h2>
    <ul>
        <li v-for="(value,k) in car" :key="k">
            {{k}}-{{value}}
        </li>
    </ul>

    <!-- 遍历字符串 -->
    <h2>测试遍历字符串(用得少)</h2>
    <ul>
        <li v-for="(char,index) in str" :key="index">
            {{char}}-{{index}}
        </li>
    </ul>

    <!-- 遍历指定次数 -->
    <h2>测试遍历指定次数(用得少)</h2>
    <ul>
        <li v-for="(number,index) in 5" :key="index">
            {{index}}-{{number}}
        </li>
    </ul>
</div>

<script type="text/javascript">

    new Vue({
        el:'#root',
        data:{
            persons:[
                {id:'001',name:'张三',age:18},
                {id:'002',name:'李四',age:19},
                {id:'003',name:'王五',age:20}
            ],
            car:{
                name:'奥迪A8',
                price:'70万',
                color:'黑色'
            },
            str:'hello'
        }
    })
</script>

注:
: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更新 ==> 界面效果没问题, 但**效率低**。
                               1. 如果结构中还**包含输入类的DOM**:
                                               会产生**错误DOM更新** ==> 界面有问题。
                     4. 开发中如何选择key?:
                               1.最好使用每条数据的 **唯一标识作为key, 比如id、手机号、身份证号、学号** 等唯一值。
                               2.如果不存在对数据的逆序添加、逆序删除等破坏顺序操作,仅用于渲染列表用于展示,
                                   使用 index 作为 key 是没有问题的。
                     
                     ![](https://raw.githubusercontent.com/T4mako/ImageBed/main/20230511193617.png)
                     
                     ![image-20230511193447250](https://raw.githubusercontent.com/T4mako/ImageBed/main/image-20230511193447250.png)
    
<li v-for="(p,index) of persons" :key="p.id">
    {{p.name}}-{{p.age}}
    <input type="text">
</li>

1.11.2、列表过滤与排序

通过 vue 对数据实现关键字 查找排序

列表过滤

<div id="root">
    <h2>人员列表</h2>
    <input type="text" placeholder="请输入名字" v-model="keyWord">
    <ul>
        <li v-for="(p,index) of filPerons" :key="index">
            {{p.name}}-{{p.age}}-{{p.sex}}
        </li>
    </ul>
</div>

<script type="text/javascript">
    //用 watch 实现
    /* new Vue({
				el:'#root',
				data:{
					keyWord:'',
					persons:[
						{id:'001',name:'马冬梅',age:19,sex:'女'},
						{id:'002',name:'周冬雨',age:20,sex:'女'},
						{id:'003',name:'周杰伦',age:21,sex:'男'},
						{id:'004',name:'温兆伦',age:22,sex:'男'}
					],
					filPerons:[]
				},
				watch:{
					keyWord:{
						immediate:true,
						handler(val){
							this.filPerons = this.persons.filter((p)=>{
								return p.name.indexOf(val) !== -1
							})
						}
					}
				}
			}) */

    //用computed实现
    new Vue({
        el:'#root',
        data:{
            keyWord:'',
            persons:[
                {id:'001',name:'马冬梅',age:19,sex:'女'},
                {id:'002',name:'周冬雨',age:20,sex:'女'},
                {id:'003',name:'周杰伦',age:21,sex:'男'},
                {id:'004',name:'温兆伦',age:22,sex:'男'}
            ]
        },
        computed:{
            filPerons(){
                return this.persons.filter((p)=>{
                    return p.name.indexOf(this.keyWord) !== -1
                })
            }
        }
    }) 
</script>

列表排序

<div id="root">
    <h2>人员列表</h2>
    <input type="text" placeholder="请输入名字" v-model="keyWord">
    <button @click="sortType = 2">年龄升序</button>
    <button @click="sortType = 1">年龄降序</button>
    <button @click="sortType = 0">原顺序</button>
    <ul>
        <li v-for="(p,index) of filPerons" :key="p.id">
            {{p.name}}-{{p.age}}-{{p.sex}}
            <input type="text">
        </li>
    </ul>
</div>

<script type="text/javascript">

    new Vue({
        el:'#root',
        data:{
            keyWord:'',
            sortType:0, //0原顺序 1降序 2升序
            persons:[
                {id:'001',name:'马冬梅',age:30,sex:'女'},
                {id:'002',name:'周冬雨',age:31,sex:'女'},
                {id:'003',name:'周杰伦',age:18,sex:'男'},
                {id:'004',name:'温兆伦',age:19,sex:'男'}
            ]
        },
        computed:{
            filPerons(){
                const arr = this.persons.filter((p)=>{
                    return p.name.indexOf(this.keyWord) !== -1
                })
                //判断一下是否需要排序
                if(this.sortType){
                    arr.sort((p1,p2)=>{
                        return this.sortType === 1 ? p2.age-p1.age : p1.age-p2.age
                    })
                }
                return arr
            }
        }
    }) 
</script>

1.12、Vue 检测数据原理

Vue监视数据的原理:
1、vue 会监视 data 中 所有层次(到基本数据类型,包括数组) 的数据。(添加到 _data 中,并添加 get set 方法)

​ 2、如何监测对象中的数据?
​ 通过setter实现监视,且要在new Vue时就传入要监测的数据。
​ 对象中后追加的属性,Vue默认不做响应式处理
​ 如需给后添加的属性做响应式,请使用如下API:
Vue.set(target,propertyName/index,value) 或
vm/vc.$set(target,propertyName/index,value)

​ 3、如何监测数组中的数据?
​ 通过包裹数组更新元素的方法实现,本质就是做了两件事:
​ ① 调用原生对应的方法对数组进行更新。
​ ② 重新解析模板,进而更新页面。

​ 4、在 Vue 修改数组中的某个元素一定要用如下方法:
​ 使用这些API:push()、pop()、shift()、unshift()、splice()、sort()、reverse()
​ Vue.set() 或 vm.$set()

​ 特别注意:Vue.set() 和 vm.$set() 不能给vm 或 vm 的根数据对象 添加属性!!!(第一个参数必须是 vm.XXXopen in new window 或 vm._data.XXX

1.12.1、模拟一个数据检测

Object.defineProperty()open in new window 方法

let data = {
    name:'zhangsan',
    address:'shanghai',
}

//创建一个监视的实例对象,用于监视data中属性的变化
const obs = new Observer(data)		
console.log(obs)	

//准备一个vm实例对象
let vm = {}
vm._data = data = obs

function Observer(obj){
    //汇总对象中所有的属性形成一个数组
    const keys = Object.keys(obj)
    //遍历
    keys.forEach((k)=>{
        // console.log(this);
        // this 指 Observer 对象
        Object.defineProperty(this,k,{
            get(){
                return obj[k]
            },
            set(val){
                console.log(`${k}被改了,我要去解析模板,生成虚拟DOM.....我要开始忙了`)
                obj[k] = val
            }
        })
    })
}

1.12.2、Vue.set()

Vue.set(vm.属性名,属性名/索引,属性值)
第一个参数也可以是 vm._data.属性名

const vm = new Vue({
    el:'#root',
    data:{
        school:{
            name:'尚硅谷',
            address:'北京',
        },
        student:{
            name:'tom',
            age:{
                rAge:40,
                sAge:29,
            },
            friends:[
                {name:'jerry',age:35},
                {name:'tony',age:36}
            ]
        }
    },
    methods: {
        addSex(){
            // 给 student 添加一个属性 sex
            // Vue.set(this.student,'sex','男')
            this.$set(this.student,'sex','男')
        }
    }
})

1.12.3、Vue 检测数组原理

https://cn.vuejs.org/guide/essentials/list.html#array-change-detectionopen in new window

Vue 能够侦听响应式数组的变更方法,并在它们被调用时触发相关的更新
这些变更方法包括:push()pop()shift()unshift()splice()sort()reverse()

注:通过索引更改单个元素变化不会被 Vue 监测到

<script type="text/javascript">
const vm = new Vue({
    el:'#root',
    data:{
        student:{
            name:'tom',
            age:18,
            hobby:['抽烟','喝酒','烫头'],
            friends:[
                {name:'jerry',age:35},
                {name:'tony',age:36}
            ]
        }
    },
    methods: {
        addSex(){
            // Vue.set(this.student,'sex','男')
            this.$set(this.student,'sex','男')
        },
        addFriend(){
            this.student.friends.unshift({name:'jack',age:70})
        },
        updateFirstFriendName(){
            this.student.friends[0].name = '张三'
        },
        addHobby(){
            this.student.hobby.push('学习')
        },
        updateHobby(){
            // this.student.hobby.splice(0,1,'开车')
            // Vue.set(this.student.hobby,0,'开车')
            this.$set(this.student.hobby,0,'开车')
        },
        removeSmoke(){
            this.student.hobby = this.student.hobby.filter((h)=>{
                return h !== '抽烟'
            })
        }
    }
})
</script>

1.13、收集表单数据

收集表单数据:
若 <input type="text"/>,则 v-model 收集的是 value 值,用户输入的就是value值。
若:<input type="radio"/>,则 v-model收集的是 value 值,且要给标签配置value 值。
若:<input type="checkbox"/>
1.没有配置 input 的 value 属性,那么收集的就是 checked(勾选 or 未勾选,是布尔值)
2.配置 input 的 value 属性:
① v-model 的初始值是非数组,那么收集的就是checked(勾选 or 未勾选,是布尔值)
② v-model的初始值是数组,那么收集的的就是value组成的数组

v-model的三个修饰符:
v-model.lazy:失去焦点再收集数据
v-model.number:输入字符串转为有效的数字
v-model.trim:输入首尾空格过滤

<div id="root">
    <form @submit.prevent="demo">
        账号:<input type="text" v-model.trim="userInfo.account"> <br/><br/>
        密码:<input type="password" v-model="userInfo.password"> <br/><br/>
        年龄:<input type="number" v-model.number="userInfo.age"> <br/><br/>
        性别:
        男<input type="radio" name="sex" v-model="userInfo.sex" value="male"><input type="radio" name="sex" v-model="userInfo.sex" value="female"> <br/><br/>
        爱好:
        学习<input type="checkbox" v-model="userInfo.hobby" value="study">
        打游戏<input type="checkbox" v-model="userInfo.hobby" value="game">
        吃饭<input type="checkbox" v-model="userInfo.hobby" value="eat">
        <br/><br/>
        所在地
        <select v-model="userInfo.city">
            <option value="">请选择校区</option>
            <option value="beijing">北京</option>
            <option value="shanghai">上海</option>
            <option value="shenzhen">深圳</option>
            <option value="wuhan">武汉</option>
        </select>
        <br/><br/>
        其他信息:
        <textarea v-model.lazy="userInfo.other"></textarea> <br/><br/>
        <input type="checkbox" v-model="userInfo.agree">阅读并接受<a href="http://www.atguigu.com">《用户协议》</a>
        <button>提交</button>
    </form>
</div>
<script type="text/javascript">
    new Vue({
        el:'#root',
        data:{
            userInfo:{
                account:'',
                password:'',
                age:18, //为数字
                sex:'female',
                hobby:[], //为数组
                city:'beijing',
                other:'',
                agree:''
            }
        },
        methods: {
            demo(){
                console.log(JSON.stringify(this.userInfo))
            }
        }
    })
</script>

1.14、过滤器

过滤器:
定义:对要显示的数据进行特定格式化后再显示(适用于一些简单逻辑的处理)。
语法:
1.注册过滤器:Vue.filter(name,callback) 或 new Vue{filters:{}} (全局过滤器 与 局部过滤器)
2.使用过滤器:0 或 v-bind:属性 = "xxx | 过滤器名"

​ 备注:
​ 1.过滤器也可以接收额外参数、多个过滤器也可以串联
​ 2.并没有改变原本的数据, 是产生新的对应的数据

<div id="root">
    <h2>显示格式化后的时间</h2>
    <!-- 计算属性实现 -->
    <h3>现在是:{{fmtTime}}</h3>
    <!-- methods实现 -->
    <h3>现在是:{{getFmtTime()}}</h3>
    <!-- 过滤器实现 -->
    <h3>现在是:{{time | timeFormater}}</h3>
    <!-- 过滤器实现(传参) -->
    <h3>现在是:{{time | timeFormater('YYYY_MM_DD') | mySlice}}</h3>
    <h3 :x="msg | mySlice">尚硅谷</h3>
</div>
<div id="root2">
    <h2>{{msg | mySlice}}</h2>
</div>

<script type="text/javascript">
    //全局过滤器
    Vue.filter('mySlice',function(value){
        return value.slice(0,4)
    })
    new Vue({
        el:'#root',
        data:{
            time:1621561377603, //时间戳
            msg:'你好,尚硅谷'
        },
        computed: {
            fmtTime(){
                return dayjs(this.time).format('YYYY年MM月DD日 HH:mm:ss')
            }
        },
        methods: {
            getFmtTime(){
                return dayjs(this.time).format('YYYY年MM月DD日 HH:mm:ss')
            }
        },
        //局部过滤器
        filters:{
            timeFormater(value,str='YYYY年MM月DD日 HH:mm:ss'){ //如果str有值,就传过来,没有就是默认值
                // console.log('@',value)
                // 引入外部函数 day.js
                return dayjs(value).format(str)
            }
        }
    })
    new Vue({
        el:'#root2',
        data:{
            msg:'hello,atguigu!'
        }
    })
</script>

1.15、其他内置命令

1、v-text

v-text 指令:
作用:向其所在的节点中渲染文本内容。
与插值语法的区别:v-text会完全替换掉节点中的内容,则不会。

<div v-text="str"></div> <!-- str为<h1>AAA</h1> 都会解析成字符串 -->

2、v-html

v-html 指令:
作用:向指定节点中渲染包含html结构的内容。
与插值语法的区别:
① v-html会替换掉节点中所有的内容,则不会。
② v-html可以 识别 html 结构
严重注意:v-html有安全性问题 !!!!
在网站上动态渲染任意HTML是非常危险的,容易导致XSS攻击。
一定要在可信的内容上使用v-html,永不要用在用户提交的内容上!

<div v-html="str"></div> <!-- str为<h1>AAA</h1> 解析成 AAA -->

3、v-cloak

v-cloak 指令(没有值):
本质是一个特殊属性,Vue实例 创建完毕并 接管容器 后,会 删掉v-cloak 属性。
使用 css 配合 v-cloak 可以解决网速慢时页面展示出的问题。(解决页面闪现问题)

<head>
    <style>
        /* 选中所有带 v-clock 属性的标签 */
        [v-cloak]{
            display:none;
        }
    </style>
</head>
<body>
    <div id="root">
        <h2 v-cloak>{{name}}</h2>
    </div>
</body>

4、v-once

v-once 指令:
v-once 所在节点在初次动态渲染后,就视为静态内容了。(只渲染一次
以后数据的改变不会引起 v-once 所在结构的更新,可以用于优化性能。

<h2 v-once>初始化的n值是:{{n}}</h2>

5、v-pre

v-pre指令:
跳过其所在节点的编译过程。
可利用它跳过:没有使用指令语法、没有使用插值语法的节点,会加快编译。

<h2 v-pre>当前的n值是:{{n}}</h2> <!-- 当前的n值是:{{n}} -->

1.16、自定义 Vue 指令

一、定义语法:

​ ① 局部指令:

new Vue({
    directive:{指令名:配置对象}
})

//或者

new Vue({
    directives{指令名:回调函数	}
})

​ ② 全局指令:

Vue.directive(指令名,配置对象) 
或  
Vue.directive(指令名,回调函数)

二、配置对象中常用的3个回调:

方法的两个参数:当前 dom 对象 传入的值

​ (1).bind(element,binding):指令与元素成功绑定时调用。

​ (2).inserted(element,binding):指令所在元素被插入页面时调用。

​ (3).update(element,binding):指令所在模板结构被重新解析时调用。

三、备注:
1.指令定义时不加v-,但使用时要加v-;
2.指令名如果是多个单词,要使用 kebab-case(下划线) 命名方式,不要用camelCase命名。.

<div id="root">
    <h2>当前的n值是:<span v-text="n"></span> </h2>
    <!-- <h2>放大10倍后的n值是:<span v-big-number="n"></span> </h2> -->
    <h2>放大10倍后的n值是:<span v-big="n"></span> </h2>
    <button @click="n++">点我n+1</button>
    <hr/>
    <input type="text" v-fbind:value="n">
</div>

<script type="text/javascript">

    //定义全局指令
    /* Vue.directive('fbind',{
			//指令与元素成功绑定时(一上来)
			bind(element,binding){
				element.value = binding.value
			},
			//指令所在元素被插入页面时
			inserted(element,binding){
				element.focus()
			},
			//指令所在的模板被重新解析时
			update(element,binding){
				element.value = binding.value
			}
		}) */

    new Vue({
        el:'#root',
        data:{
            n:1
        },
        directives:{
            //big函数何时会被调用?1.指令与元素成功绑定时(一上来)。2.指令所在的模板被重新解析时。
            /* 'big-number'(element,binding){
					// console.log('big')
					element.innerText = binding.value * 10
				}, */
            big(element,binding){
                console.log('big',this) //注意此处的this是window
                // console.log('big')
                element.innerText = binding.value * 10
            },
            fbind:{
                //指令与元素成功绑定时(一上来)
                bind(element,binding){
                    element.value = binding.value
                },
                //指令所在元素被插入页面时
                inserted(element,binding){
                    element.focus()
                },
                //指令所在的模板被重新解析时
                update(element,binding){
                    element.value = binding.value
                }
            }
        }
    })
</script>

1.17、Vue 生命周期

Vue 在关键时刻帮我们调用的一些特殊名称的函数
生命周期函数的名字不可更改,内容可以自由编写
生命周期函数 中的 this 指向是 vm组件实例 对象。

生命周期 分为 挂载流程 , 更新流程 ,销毁流程
生命周期的函数: beforeCreated()、created()、beforeMount()、mounted()、beforeUpdate()、upDdated()、beforeDestory()、destroyed()

常用的生命周期钩子:
mounted: 发送ajax请求、启动定时器、绑定自定义事件、订阅消息等【初始化操作】。
beforeDestroy: 清除定时器、解绑自定义事件、取消订阅消息等【收尾工作】。

mounted 的执行时期:Vue 完成模板的解析并把初始的真实DOM元素放入页面后(挂载完毕)调用 mounted
(Vue-dom流程:生成虚拟dom -> 虚拟dom转换为真实dom -> dom挂载到页面上)

关于销毁Vue实例
销毁后借助Vue开发者工具看不到任何信息。
销毁后 自定义事件 会失效,但 原生DOM事件 依然有效。
一般不会在 beforeDestroy 操作数据,因为即便操作数据,也不会再触发更新流程了。

生命周期

评论
  • 按正序
  • 按倒序
  • 按热度
Powered by Waline v2.15.5