Vue3

Vue3启程

关键字:创建vue实例、响应式、响应式原理、组合式API

1. 初始Vue3

Vue2存在一些缺陷,Vue3解决了这些缺陷,剔除了过滤器,保留了绝大部分的功能,可以说更加好用,更加丰富。
我们将Vue2和Vue3对比一下,就很直观了:

<div id="app">
    <h2>姓名:{{name}}</h2>
    <h2>年龄:{{age}}</h2>
    <button @click="updateData">修改数据</button>
</div>

Vue2中--创建实例的方式

new Vue({
    //指定挂载容器
    // el:'#app',
    //定义属性
    data() {
        return {
            name:'张三',
            age:20
        }
    },
    //定义方法
    methods: {
        updateData(){
            this.name = '李四'
            this.age = 25
        }
    },
}).$mount('#app')  //指定当前vue实例挂载的容器 

Vue3中--创建实例的方式

// Vue3中--创建实例的方式
Vue.createApp({
    //注意:这个配置对象里面除了不能写el选项,之前怎么写,现在还可以怎么写
    //定义属性
    data() {
        return {
            name:'张三',
            age:20
        }
    },
    //定义方法
    methods: {
        updateData(){
            this.name = '李四'
            this.age = 25
        }
    },
}).mount('#app')   //只能通过mount方法指定挂载的容器,不用通过el选项指定

2. Vue2和Vue3的响应式

<div id="app">
    <h2>学生:{{stu}}</h2>
    <h2>食物:{{foods}}</h2>
    <div>
        <button @click="updateStuName">修改学生姓名</button>
        <button @click="addStuSex">添加学生性别</button>
        <button @click="delStuAge">删除学生年龄</button>
        <button @click="updateFoods2">修改第二个食物</button>
    </div>
</div>

2. Vue2和Vue3的响应式

<div id="app">
    <h2>学生:{{stu}}</h2>
    <h2>食物:{{foods}}</h2>
    <div>
        <button @click="updateStuName">修改学生姓名</button>
        <button @click="addStuSex">添加学生性别</button>
        <button @click="delStuAge">删除学生年龄</button>
        <button @click="updateFoods2">修改第二个食物</button>
    </div>
</div>
// Vue2
new Vue({
    data() {
        return {
            //学生对象
            stu:{
                name:'张三',
                age:20
            },
            //食物数组
            foods:['榴莲','葡萄','香蕉']
        }
    },
    methods: {
        updateStuName(){
            this.stu.name = '李四'
        },
        addStuSex(){
            // 直接给对象添加的属性,不具备响应式
            // this.stu.sex = '男'
            // 如果要给对象添加属性,并且添加的属性也要具备响应式,要使用$set方法
            // 方法的第一个参数是指定的对象,第二个参数是属性名,第三个参数是属性值。
            this.$set(this.stu,'sex','男')
        },
        delStuAge(){
            // 直接删除对象身上的属性,是不具备响应式的
            // delete this.stu.age
            // 如果要删除对象身上的属性,并且还要具备响应式,要使用$delete方法
            // 方法的第一个参数是指定的对象,第二个参数是属性名
            this.$delete(this.stu,'age')
        },
        updateFoods2(){
            // 直接根据索引修改数组元素,不具备响应式
            // this.foods[1] = '西瓜'
            // 操作数组中的元素,并且还要具备响应式,只能使用数组的以下方法:
            // push unshift pop shift splice reverse sort
            // this.foods.splice(1,1,'西瓜')
            // 如果就是想通过下标去操作数组,还要具备响应式,使用$set方法
            this.$set(this.foods,1,'西瓜')
        }
    },
}).$mount('#app') 
// 总结Vue2的响应式:不能直接给对象添加属性,删除对象的属性,不能直接操作数组的下标,
// 但是,Vue2同时也提供了解决这些问题的方案。

// Vue3
Vue.createApp({
    data() {
        return {
            //学生对象
            stu:{
                name:'张三',
                age:20
            },
            //食物数组
            foods:['榴莲','葡萄','香蕉']
        }
    },
    methods: {
        updateStuName(){
            this.stu.name = '李四'
        },
        addStuSex(){
            // 在Vue3中,直接给对象添加属性,新的属性依然具备响应式
            this.stu.sex = '男'
        },
        delStuAge(){
            // 在Vue3中,直接删除对象的属性,依然具备响应式
            delete this.stu.age
        },
        updateFoods2(){
            // 在Vue3中,根据下标操作数组,依然具备响应式
            this.foods[1] = '西瓜'
        }
    },
}).mount('#app')
// 总结Vue3的响应式:解决了再Vue2中的所有问题。

3. Vue2和Vue3的响应式原理

<h2 id="name"></h2>
<h2 id="age"></h2>
// Vue2的响应式原理:
// 这里的obj是源对象
let obj = {
    name:'张三',
    age:20
}
// 在页面中显示姓名和年龄
document.getElementById('name').innerText = obj.name
document.getElementById('age').innerText = obj.age
// 这里的obj2代理对象---由obj2代理obj
let obj2 = {}
// 给obj2定义name属性
Object.defineProperty(obj2,'name',{
    get(){
        return obj.name
    },
    set(value){
        obj.name = value
        document.getElementById('name').innerText = obj.name
    }
})
// 给obj2定义age属性
Object.defineProperty(obj2,'age',{
    get(){
        return obj.age
    },
    set(value){
        obj.age = value
        document.getElementById('age').innerText = obj.age
    }
}) 

// Vue3的响应式原理:
// 这里的obj是源对象
let obj = {
    name:'张三',
    age:20
}
// 在页面中显示姓名和年龄
document.getElementById('name').innerText = obj.name
document.getElementById('age').innerText = obj.age
// 这里的obj2代理对象---由obj2代理obj
// new Proxy(源对象,{...})的方式,创建代理对象
let obj2 = new Proxy(obj,{
    //读取属性,参数分别是:源对象,属性名
    get(target, property){
        // 直接根据源对象返回源对象身上的属性
        // return target[property]
        // 通过发射对象,发射输出源对象身上的属性
        return Reflect.get(target,property)
    },
    //设置属性,参数分别是:源对象,属性名,属性值
    set(target, property,value){
        // target[property] = value
        if(Reflect.has(target,property)){
            Reflect.set(target, property,value)
            document.getElementById(`${property}`).innerText = value
        }
    },
    //删除属性,参数分别是:源对象,属性名
    deleteProperty(target, property){
        // delete target[property]
        Reflect.deleteProperty(target, property)
    }
})

4. 引出Vue3新推出的组合式API

这是很棒的一个功能,和另一个R框架有点像,让数据和方法在一起,方便维护。

<div id="app">
    <div>
        <h2>学生信息</h2>
        <!-- 注意:ref对象在模板只不需要.value的方式获取里面的值 -->
        <h4>姓名:{{stuName}}</h4>
        <h4>年龄:{{stuAge}}</h4>
        <button @click="updateStu">修改学生信息</button>
    </div>
    <div>
        <h2>汽车信息</h2>
        <h4>车名:{{carName}}</h4>
        <h4>车价:{{carPrice}}</h4>
        <button @click="updateCar">修改汽车信息</button>
    </div>
    <div>
        <h2>手机信息</h2>
        <h4>名称:{{phoneName}}</h4>
        <h4>颜色:{{phoneColor}}</h4>
        <button @click="updatePhone">修改手机信息</button>
    </div>
    <div>
        <h2>食物信息</h2>
        <h4>名称:{{foodName}}</h4>
        <h4>价格:{{foodPrice}}</h4>
        <button @click="updateFood">修改食物信息</button>
    </div>
</div>

什么是组合式API(Composition API),就是Vue推出的一些新的方法,这个方法在setup中使用, 从Vue身上获取ref组合式API函数


let {ref} = Vue
Vue.createApp({
    // 注意:Vue2中,Vue实例的data选项可以是一个对象,也可以是一个方法,由方法返回一个对象
    // 但是,组件中data选项必须是一个方法。
    // Vue3中,无论是Vue实例,还是组件,data选项都必须是一个方法。
    // 我们之前习惯将所有的数据放在data选项中定义,所有的方法放在methods选项中定义,
    // 所有的计算属性放在computed选项中定义,所有的侦听器放在watch选项中定义,
    // 这样就会导致一个业务的代码会拆分到多个结构中去写,如果一个页面中要操作很多个业务,代码后期维护成本会很高。
    // 所以,Vue3引入了组合式API,简化之前繁琐的过程,将相同业务的代码靠在一起写。
    /* data: function () {
        return {
            //定义学生数据
            stuName: '张三',
            stuAge: '20',
            //汽车信息
            carName: '奔驰',
            carPrice: '50W',
            //手机信息
            phoneName: 'iphone',
            phoneColor: '白色',
            //食物信息
            foodName: '汉堡',
            foodPrice: '¥20'
        }
    }, 
    methods: {
        //修改学生的方法
        updateStu(){
            this.stuName = '李四'
            this.stuAge = 30
        },
        //修改汽车的方法
        updateCar(){
            this.carName = '宝马'
            this.carPrice = '40W'
        },
        //修改手机的方法
        updatePhone(){
            this.phoneName = '华为'
            this.phoneColor = '蓝色'
        },
        updateFood(){
            this.foodName = '蛋糕'
            this.foodPrice = '¥30'
        }
    }, */
    // setup方法是所有组合式API的入口
    setup() {
        // 定义学生的信息
        // 在setup中,直接定义的数据是不具备响应式的,
        // 如果要使数据具备响应式,需要使用ref组合式API对数据进行包装,包装后返回的是ref对象
        let stuName = ref('张三')
        let stuAge = ref('20')
        let updateStu = () => {
            //ref对象的value属性保存的是值
            stuName.value = '李四'
            stuAge.value = 30
        }
        // 定义汽车的信息
        let carName = ref('奔驰')
        let carPrice = ref('50W')
        let updateCar = () => {
            carName.value = '宝马'
            carPrice.value = '40W'
        }
        // 定义手机的信息
        let phoneName = ref('iphone')
        let phoneColor = ref('白色')
        let updatePhone = () => {
            phoneName.value = '华为'
            phoneColor.value = '蓝色'
        }
        // 定义食物的信息
        let foodName = ref('汉堡')
        let foodPrice = ref('¥20')
        let updateFood = () => {
            foodName.value = '蛋糕'
            foodPrice.value = '¥30'
        }

        //返回模板中需要使用的数据
        return{
            stuName,
            stuAge,
            updateStu,
            carName,
            carPrice,
            updateCar,
            phoneName,
            phoneColor,
            updatePhone,
            foodName,
            foodPrice,
            updateFood
        }
    }
}).mount('#app')
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,088评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,715评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,361评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,099评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 60,987评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,063评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,486评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,175评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,440评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,518评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,305评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,190评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,550评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,880评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,152评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,451评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,637评论 2 335

推荐阅读更多精彩内容