Vue 核心
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:单项数据绑定

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():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 为当前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
computed 与 watch 配置项
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 的原理:
虚拟 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 是没有问题的。  
<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.XXX 或 vm._data.XXX)
1.12.1、模拟一个数据检测
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-detection
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 操作数据,因为即便操作数据,也不会再触发更新流程了。
