Vue3启程

vue3官网:https://v3.cn.vuejs.org/guide/introduction.html

1. 初始Vue3

导入vue3
<script src="https://unpkg.com/vue@next"></script>

vue3创建一个vue实例
在vue3里面Vue是一个对象,通过该对象的createApp()方法,创建一个Vue实例

 Vue.createApp({
            // 注意:vue3中,取消了el选项
            // el:'#app',
            // 注意:vue3中,无论是组件和是vue实例,data选项都必须是一个方法,由方法返回对象。
            data(){
                return {
                    name:'Vue3',
                    age:'2'
                }
            }
        }).mount('#app')

在vue2里面的Vue是一个构造函数,通过该构造函数创建一个Vue实例

 new Vue({
            // el:'#app',
            // 在vue2中,data选项可以是对象,也可以是方法返回一个对象
            data:{
                name:'Vue2',
                age:'6'
            }
        }).$mount('#app') 

在vue2可以通过el选项指定一个挂载的容器,也可以通过$mount()方法指定挂载的容器
在vue3中,创建完一个组件实例,我们需要调用mount()方法将组件实例挂载到页面中

2. Vue2和Vue3的响应式

在Vue2中,Vue实例在初始化的时候,会将obj对象身上的所有数据做响应式处理。所谓响应式,指的是数据发生变化后,页面自动更新。
Vue2的响应式:不能直接给对象添加属性,删除对象的属性,不能直接操作数组的下标,但是,Vue2同时也提供了解决这些问题的方案。

 <div id="app" v-cloak>
        <div>学生信息:{{student}}</div>
        <button @click="student.name+='!'">修改学生姓名</button>
        <button @click="student.age++">修改学生年龄</button>
        <button @click="addSex">添加性别</button>
        <button @click="delName">删除姓名</button>
        <div>食物:{{foods}}</div>
        <button @click="addFood">添加食物</button>
        <button @click="delFood">删除帝王蟹</button>
    </div>
// vue2的响应式
        new Vue({
            el:'#app',
            data:{
                student:{
                    name:'张三',
                    age:20
                },
                foods:['鱼翅','松茸','鱼子酱','帝王蟹','熊掌']
            },
            methods: {
                //添加性别
                addSex(){
                    //后添加的属性是非响应式的
                    // this.student.sex='男'
                    //可以通过$forceUpdate()强制页面更新一次
                    // this.$forceUpdate()

                    //推荐使用$set方法给对象添加新的属性,确保新添加的属性同样具备响应式
                    this.$set(this.student,'sex','男')
                },
                //删除姓名
                delName(){
                    // 直接使用delete方式删除对象的属性后,不具备响应式
                    // delete this.student.name
                    //使用$delete方法,删除对象的属性后,继续具备响应式
                    this.$delete(this.student,'name')
                },
                //添加食物
                addFood(){
                    // 操作数组后同时要具有响应式,必须要使用下面的方法:
                    // push pop unshift shift sort reverse splice
                    // this.foods.push('佛跳墙')
                    // this.foods[5] = '佛跳墙'
                    // this.$forceUpdate()
                    //推荐使用$set方法根据下标添加数组元素,确保新添加的元素同样具备响应式
                    this.$set(this.foods,5,'佛跳墙')
                },
                //删除食物
                delFood(){
                    // this.foods.splice(3,1)
                    //直接根据下标删除数组元素,不具备响应式
                    // this.foods[3] = null

                    //使用$delete方法,删除数组中指定位置的元素,继续具备响应式
                    this.$delete(this.foods,3)
                }
            },
        }) 

vue3修复了vue2中响应式的所有缺陷。在Vue3中,直接给对象添加属性、直接删除对象的属性、根据下标操作数组,都依然具备响应式。

Vue.createApp({
            data() {
                return {
                    student:{
                        name:'张三',
                        age:20
                    },
                    foods:['鱼翅','松茸','鱼子酱','帝王蟹','熊掌']
                }
            },
            methods: {
                addSex(){
                    this.student.sex = '男'
                },
                delName(){
                    delete this.student.name
                },
                addFood(){
                    this.foods[5] = '佛跳墙'
                },
                delFood(){
                    delete this.foods[3]
                }
            },
        }).mount("#app")

3. Vue2和Vue3的响应式原理

Vue2在实例化时,会将data里面的所有数据采用 Object.defineProperty 进行处理,从而实现响应式功能。但是你之后往data里面添加的数据,由于没有采用 Object.defineProperty 进行处理,所以不具备响应式。$set()方法,内部就是对单个属性重新采用 Object.defineProperty 进行处理,从而具备响应式。

<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

什么是组合式API(Composition API),就是Vue推出的一些新的方法,这个方法在setup中使用
setup方法是所有组合式API的入口
Vue3中,无论是Vue实例,还是组件,data选项都必须是一个方法。
我们之前习惯将所有的数据放在data选项中定义,所有的方法放在methods选项中定义,
所有的计算属性放在computed选项中定义,所有的侦听器放在watch选项中定义,
这样就会导致一个业务的代码会拆分到多个结构中去写,如果一个页面中要操作很多个业务,代码后期维护成本会很高。
所以,Vue3引入了组合式API,简化之前繁琐的过程,将相同业务的代码靠在一起写。

 // 组合式api的作用是:将原来分散开来定义的数据,方法,计算属性,监听器,组合起来定义一个完整的业务。
 // ref用于定义响应式数据
        let {ref} = Vue

        Vue.createApp({
            // setup是组合式api的舞台,所有的组合式api都要在setup里面使用
            setup() {
                // 在setup中,直接定义的数据是不具备响应式的,
                // 如果要使数据具备响应式,需要使用ref组合式API对数据进行包装,包装后返回的是ref对象
                //定义汽车相关数据
                // 使用ref()方法,定义一个响应式对象
                let carName=ref('保时捷')
                let carPrice=ref('100W')
                //定义汽车相关方法
                function updateCar(){
                    //修改对象的值,要通过value属性
                    carName.value = '特斯拉'
                    carPrice.value = '80W'
                }
                //定义飞机相关数据
                let planeName=ref('波音747')
                let planePrice=ref('10Y')
                //定义飞机相关方法
                function updatePlane(){
                    planeName.value = 'B52轰炸机',
                    planePrice.value = '30Y'
                }
                //手表
                let watchName=ref('劳力士')
                let watchPrice=ref('10W')
                function updateWacth(){
                    watchName.value = '欧米伽',
                    watchPrice.value = '4W'
                }
                //手机
                let phoneName=ref('iphone13')
                let phonePrice=ref('5999')
                function updatePhone(){
                    phoneName.value = '华为',
                    phonePrice.value = '6999'
                }
                return{
                    //返回汽车相关数据
                    carName,
                    carPrice,
                    updateCar,
                    //返回飞机相关数据
                    planeName,
                    planePrice,
                    updatePlane,
                    //返回手表相关数据
                    watchName,
                    watchPrice,
                    updateWacth,
                    //返回手机相关数据
                    phoneName,
                    phonePrice,
                    updatePhone
                }
            },
 }).mount('#app')

5. ref和reactive

ref 和 reactive 都是用于定义响应式数据。通常情况下,基本类型的数据,选择用ref定义;引用类型的数据,选择用reactive定义。
ref方法:返回的是ref对象,ref对象的value属性是一个代理对象(Proxy)。使用ref既可以定义基本类型数据,也可以定义引用类型数据。注意:修改ref对象的值,必须要先.value再.具体的属性。
reactive方法:直接返回一个代理对象(Proxy)。reactive只能定义引用类型数据.

<div id="app">
    <h4>姓名:{{name}}</h4>
    <h4>学生:{{stu}}</h4>
    <button @click="updateName">修改姓名</button>
    <button @click="updateStu">修改学生</button>
</div>
let {ref,reactive} = Vue
Vue.createApp({
    setup() {
        let name = ref('张三')
        let updateName = ()=>{
            name.value = '张杰'
        }
        /* let stu = ref({
            name:'李四',
            age:20
        })
        let updateStu = ()=>{
            // 注意:修改ref对象的值,每次都要先点value
            stu.value.name = '李明'
            stu.value.age = 30
        } */
        // reactive组合式API方法,根据源对象返回一个代理对象(Proxy对象)
        let stu = reactive({
            name:'李四',
            age:20
        })
        let updateStu = ()=>{
            // Proxy对象,不需要先点value
            stu.name = '李明'
            stu.age = 30
        }
        return {
            name,
            updateName,
            stu,
            updateStu
        }
    }
}).mount('#app')
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,236评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,867评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,715评论 0 340
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,899评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,895评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,733评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,085评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,722评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,025评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,696评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,816评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,447评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,057评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,009评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,254评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,204评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,561评论 2 343

推荐阅读更多精彩内容