Vue2(生命周期,列表排序,计算属性和监听器)

news2024/11/26 6:19:33

目录

      • 前言
      • 一,生命周期
        • 1.1,生命周期函数简介
        • 1.2,Vue的初始化流程
        • 1.3,Vue的更新流程
        • 1.4, Vue的销毁流程
        • 1.5, 回顾生命周期
        • 1.,6,代码演示
          • 1.6-1,beforeCreate
          • 1.6-2,created
          • 1.6-3,beforeMount
          • 1.6-4,mounted
          • 1.6-5,beforeUpdate
          • 1.6-6,updated
        • 1.7,生命周期函数的使用场景
      • 二,vue中列表排序
        • 2.1,vue中数据更新的问题
          • 2.1-1,对象新增数据更新问题
          • 2.1-1,数组数据更新问题
      • 三,计算属性computed
      • 四,监视属性watch
      • 最后

前言

上一章博客我们讲解了Vue基础
这一章我们来讲Vue生命周期,列表过滤,计算属性和监听器

一,生命周期

  • 通常也叫生命周期回调函数、生命周期函数、生命周期钩子
  • vue初始化时在不同的阶段调用的不同函数
  • 生命周期函数的this指向为vue实例,名字不能更改

1.1,生命周期函数简介

“生命周期”指实例对象从构造函数开始执行(被创建)到被GC(Garbage Collection)回收销毁的整个存在的时期
生命周期函数是为了在实例对象的各个阶段方便开发者控制而产生,在不同的时期有不同的生命周期函数,可以利用不同时期的生命周期函数去完成不同的操作

实际的开发过程中,生命周期函数的使用较为频繁,需要重点掌

1.2,Vue的初始化流程

在这里插入图片描述

  • beforeCreate阶段
    • vue中data的数据和methods的方法还不能使用
  • created阶段
    • vue中data的数据和methods的方法已可以使用
  • beforeMount阶段
    • 页面可以展示内容,但是是未编译,最终都是不能操作的DOM结构,展示时间短
  • mounted
    • 页面显示编译后的DOM
    • vue的初始化过程结束
    • 此阶段可进行:定时器、网络请求、订阅消息、绑定事件等

1.3,Vue的更新流程

在这里插入图片描述

  • beforeUpdate
    • 数据是新的,页面还没有更新
  • Updated
    • 数据是新的,页面同步更新

1.4, Vue的销毁流程

在这里插入图片描述

  • beforeDestroy
    • 此阶段可关闭定时器和取消订阅
    • 数据、方法可以访问但是不触发更新
  • destroy
    • 一切都结束了

1.5, 回顾生命周期

  • 生命周期函数
    • 创建前、创建后(beforeCreate、created)
    • 挂载前、挂载后(beforeMount、mounted)
    • 更新前、更新后(beforeUpdate、updated)
    • 销毁前、销毁后(beforeDestroy、destroyed)
  • 常用的生命周期函数
    • mounted
      • 开启定时器
      • 发送ajax请求
      • 订阅消息
      • 绑定自定义事件
    • beforeDestroy
      • 清除定时器
      • 取消订阅、事件监听
      • 解绑自定义事件
  • vue实例销毁
    • vue开发者工具的数据为空
    • 销毁后自定义事件失效
    • 不要在beforeDestroy进行数据的操作,不会再走更新流程

1.,6,代码演示

1.6-1,beforeCreate
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>
<body>
    <div id="app">
        {{myName}}
    </div>
</body>
<script src="../js/vue2.7.js"></script>
<script>
    var app = new Vue({
        el:"#app",
        data(){
            return{
                myName:"abc"
            }
        },
        beforeCreate(){
            var bobyDom = document.getElementsByTagName("body")[0].innerHTML
            // beforeCreate中,data的数据是没有被定义的
            console.log("beforeCreate",this.myName,bobyDom);
        },
    })
</script>
</html>

在这里插入图片描述

1.6-2,created
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>
<body>
    <div id="app">
        {{myName}}
    </div>
</body>
<script src="../js/vue2.7.js"></script>
<script>
    var app = new Vue({
        el:"#app",
        data(){
            return{
                myName:"abc"
            }
        },
        created(){
            var bobyDom = document.getElementsByTagName("body")[0].innerHTML
            // 做一下页面的数据初始化工作。比如说发起ajax请求
            console.log("created",this.myName,bobyDom);
        },
    })
</script>
</html>

在这里插入图片描述

1.6-3,beforeMount
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>
<body>
    <div id="app">
        {{myName}}
    </div>
</body>
<script src="../js/vue2.7.js"></script>
<script>
    var app = new Vue({
        el:"#app",
        data(){
            return{
                myName:"abc"
            }
        },
        beforeMount(){
            var bobyDom = document.getElementsByTagName("body")[0].innerHTML
            
            console.log("beforeMount",this.myName,bobyDom);
        },
    })
</script>
</html>

在这里插入图片描述

1.6-4,mounted
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>
<body>
    <div id="app">
        {{myName}}
    </div>
</body>
<script src="../js/vue2.7.js"></script>
<script>
    var app = new Vue({
        el:"#app",
        data(){
            return{
                myName:"abc"
            }
        },
        mounted(){
            var bobyDom = document.getElementsByTagName("body")[0].innerHTML
            // 数据已经渲染到View中
            console.log("mounted",this.myName,bobyDom);
        },
    })
</script>
</html>

在这里插入图片描述

1.6-5,beforeUpdate
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>
<body>
    <div id="app">
        {{myName}}
    </div>
</body>
<script src="../js/vue2.7.js"></script>
<script>
    var app = new Vue({
        el:"#app",
        data(){
            return{
                myName:"abc"
            }
        },
        beforeUpdate(){
            var bobyDom = document.getElementsByTagName("body")[0].innerHTML
            // 数据更新前,数据未改变
            console.log("beforeUpdate",this.myName,bobyDom);
        },
    })
</script>
</html>

在这里插入图片描述

1.6-6,updated
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>
<body>
    <div id="app">
        {{myName}}
    </div>
</body>
<script src="../js/vue2.7.js"></script>
<script>
    var app = new Vue({
        el:"#app",
        data(){
            return{
                myName:"abc"
            }
        },
        updated(){
            var bobyDom = document.getElementsByTagName("body")[0].innerHTML
            // 数据更新后,数据已改变
            console.log("updated",this.myName,bobyDom);
        }
    })
</script>
</html>

在这里插入图片描述

1.7,生命周期函数的使用场景

  • 在beforeCreate生命周期函数运行时,可以添加loading动画
  • 在created生命周期函数运行时,可以结束loading动画,还可以做一些初始化,实现函数自执行等操作
  • 最经常使用的是mounted生命周期函数
    • 可以发起后端数据请求,取回数据
    • 可以接收页面之间传递的参数
    • 可以子组件向父组件传递参数等

二,vue中列表排序

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>
<body>
    <div id="app">
        <div>
            <input type="text" v-model="searchInput" name="" id="">
            <button @click="paixu(1)">升序</button>
            <button @click="paixu(2)">降序</button>
            <button @click="paixu(3)">原序列</button>
            <div v-for="(item,index) in searchList">
                {{item.name}}--{{item.price}}
            </div>
        </div>
    </div>
</body>
<script src="../js/vue2.7.js"></script>
<script>
    var app = new Vue({
        el:'#app',
        data(){
            return{
                keyword:0, // 用来排序
                searchInput:"",
                goodList:[
                    {name:'牛仔裤',price:200},
                    {name:'运动鞋',price:100},
                    {name:'跑步鞋',price:300},
                    {name:'篮球鞋',price:400},
                ],
            }
        },
        methods:{
            paixu(val){
                this.keyword=val
            }
        },
        computed:{
            searchList(){
                // 把goodList的内容过滤,把符合条件的形成一个新的数组
                // filter是数字的过滤方法,这个方法的参数是一个函数,此函数返回true,则把此项返回新的数组中
                let list = this.goodList.filter((item)=>{
                    return item.name.indexOf(this.searchInput)!==-1
                })

                if(this.keyword){
                    list.sort((a1,a2)=>{
                        return this.keyword === 1
                        ? a1.price - a2.price
                        : a2.price - a1.price
                    })
                }

                return list
            }
        }
    })
</script>
</html>

在这里插入图片描述

2.1,vue中数据更新的问题

2.1-1,对象新增数据更新问题
  • 描述
    • 通过普通对象添加属性方法,Vue不能监测到且不是响应式
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>
<body>
    <div id="app">
        <div v-for="(attr,key) in person">
            {{key}} -- {{attr}}
        </div>
    </div>
</body>
<script src="../js/vue2.7.js"></script>
<script>
    var app = new Vue({
        el:'#app',
        data(){
            return{
                person:{
                    name:'张三',
                    sex:'男',
                }
            }
        },
    
    })
</script>
</html>

在这里插入图片描述
这样添加它没有没有跟对象一样封装有监测数据变化的getter、setter

  • 解决
    • Vue.set() / this.$set
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>
<body>
    <div id="app">
        <div v-for="(attr,key) in person">
            {{key}} -- {{attr}}
        </div>
    </div>
</body>
<script src="../js/vue2.7.js"></script>
<script>
    var app = new Vue({
        el:'#app',
        data(){
            return{
                person:{
                    name:'张三',
                    sex:'男',
                }
            }
        },
    
    })
</script>
</html>

在这里插入图片描述

  • 注意
    • this.$set不能给vue实例的根数据对象添加属性
2.1-1,数组数据更新问题
  • 描述
    • 直接通过数组索引值改变数组的数据,Vue监测不到改变
    • 实际在 js 内存已经把数据的第一项数据修改了
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>
<body>
    <div id="app">
        <div v-for="(attr,key) in person">
            {{key}} -- {{attr}}
        </div>
        <div>
            <input type="text" v-model="searchInput" name="" id="">
            <button @click="paixu(1)">升序</button>
            <button @click="paixu(2)">降序</button>
            <button @click="paixu(3)">原序列</button>
            <div v-for="(item,index) in searchList">
                {{item.name}}--{{item.price}}
            </div>
        </div>
    </div>
</body>
<script src="../js/vue2.7.js"></script>
<script>
    var app = new Vue({
        el:'#app',
        data(){
            return{
                keyword:0, // 用来排序
                searchInput:"",
                goodList:[
                    {name:'牛仔裤',price:200},
                    {name:'运动鞋',price:100},
                    {name:'跑步鞋',price:300},
                    {name:'篮球鞋',price:400},
                ],
                person:{
                    name:'张三',
                    sex:'男',
                }
            }
        },
        methods:{
            paixu(val){
                this.keyword=val
            }
        },
        computed:{
            searchList(){
                // 把goodList的内容过滤,把符合条件的形成一个新的数组
                // filter是数字的过滤方法,这个方法的参数是一个函数,此函数返回true,则把此项返回新的数组中
                let list = this.goodList.filter((item)=>{
                    return item.name.indexOf(this.searchInput)!==-1
                })

                if(this.keyword){
                    list.sort((a1,a2)=>{
                        return this.keyword === 1
                        ? a1.price - a2.price
                        : a2.price - a1.price
                    })
                }

                return list
            }
        }
    })
</script>
</html>

在这里插入图片描述

  • 原因
    • 因为在vue中数组并没有跟对象一样封装有监测数据变化的getter、setter
  • 解决
    • Vue在数组的原始操作方法上包裹了重新解析模板的方法,也就是说我们在data里面的数组操作方法不是原生的,是vue封装过的
    • 哪些数组操作方法经过了封装?
push() 向数组的末尾添加一个或更多元素,并返回新的长度。
pop() 删除数组的最后一个元素并返回删除的元素
shift()  删除并返回数组的第一个元素
unshift()  向数组的开头添加一个或更多元素,并返回新的长度。
splice()  从数组中添加或删除元素。
sort()   对数组的元素进行排序
reverse() 反转数组的元素顺序。

三,计算属性computed

计算属性,方法名可以直接在对象中使用.这个属性是通过计算得出的。
这个方法中的任意属性改变,都会触发这个方法
使用场景:希望把一些计算的业务逻辑放在方法中,例如:全名计算、地址计算、购物车合计

下面的示例,我们使用了三种方式:

  • 使用方法实现
  • 使用vue的computed读写方式实现
  • 使用vue的computed只读方式实现

注意:推荐使用computed的方式。有缓存机制。在页面重复调用多次的情况下,只执行一次

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>
<body>
    <div id="app">
        <div>
            <input type="text" v-model="firstName" ><br>
            <input type="text" v-model="lastName" ><br>
            <input type="text" v-model="getFullName()" ><br>
            <input type="text" v-model="fullNameReadOnly" ><br>
            <input type="text" v-model="funName" ><br>
        </div>
    </div>
</body>
<script src="../js/vue2.7.js"></script>
<script>
    var app = new Vue({
        el:"#app",
        data(){
            return{
                firstName:'科比',
                lastName:'布莱恩特',
            }
        },
        methods:{
            getFullName(){
                return this.firstName + "-" + this.lastName
            }
        },
        computed:{
            // 只读方式的简写
            fullNameReadOnly(){
                return this.firstName + "-" + this.lastName
            },
            funName:{
                get(){
                    return this.firstName + "-" + this.lastName
                },
                set(value){
                    this.firstName = value.split("-")[0]
                    this.lastName = value.split("-")[1]
                }
            }
        },
    })
</script>
</html>

在这里插入图片描述

四,监视属性watch

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>
<body>
    <div id="app">
        
    </div>
</body>
<script src="../js/vue2.7.js"></script>
<script>
    var app = new Vue({
        el:"#app",
        data(){
            return{
                isSunny:true,
                person:{
                    age:18,
                    name:'王导',
                }
            }
        },
        watch:{
            // 表示要对isSunny这个属性进行监听
            isSunny(newVal,oldVal){
                console.log("改变了",oldVal,newVal);
            },

            // 深度监听可以用来监听整个对象的改变,但要慎重使用,因为消耗性能 
            person:{
                deep:true,
                handler(oldVal,newVal){
                    console.log("改变了",oldVal,newVal);
                }
            },
            "person.name"(oldVal,newVal){
                    console.log("我也监听了");
            }
        }
    })
</script>
</html>

在这里插入图片描述

最后

送大家一句话: 世界上的好东西都是抢来的,只有弱者才会坐等分配

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/848040.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

雨水旋流过滤器、旋流雨水过滤器、水力旋流雨水过滤器、旋流分离器、旋流沉砂一体机、旋流沉砂井

产品组成 主要材料&#xff1a;PE304不锈钢 组成&#xff1a;过滤器筒体、桶盖、截止阀&#xff08;选配&#xff09;、不锈钢滤网 工作原理 雨水由过滤器进水口进入时&#xff0c;水流被引导沿过滤器内壁切线方向进入筒体。在水力、重力等作用下&#xff0c;形成雨水紧贴过…

【基础类】—前端算法类

一、排序 1. 排序方法列表 2. 常见排序方法 快速排序选择排序希尔排序 二、堆栈、队列、链表 堆栈、队列、链表 三、递归 递归 四、波兰式和逆波兰式 理论源码

Unity之ShaderGraph 节点介绍 Utility节点

Utility 逻辑All&#xff08;所有分量都不为零&#xff0c;返回 true&#xff09;Any&#xff08;任何分量不为零&#xff0c;返回 true&#xff09;And&#xff08;A 和 B 均为 true&#xff09;Branch&#xff08;动态分支&#xff09;Comparison&#xff08;两个输入值 A 和…

wm8960没有声音

最近在imx6ull上调试这个声卡&#xff0c;用官方的镜像是能发声的&#xff0c;换到自己做的镜像上&#xff0c;就没有声音。 记录一下过程&#xff1a; 内核和设备树。只要有下面的显示&#xff0c;就说明加载成功。 再看一下aplay的显示 到此&#xff0c;驱动都是正常的。但…

每天一道leetcode:剑指 Offer 32 - II. 从上到下打印二叉树 II(适合初学者)

今日份题目&#xff1a; 从上到下按层打印二叉树&#xff0c;同一层的节点按从左到右的顺序打印&#xff0c;每一层打印到一行。 示例 例如: 给定二叉树: [3,9,20,null,null,15,7], 3 / \ 9 20 / \ 15 7 返回其层次遍历结果&#xff1a; [ [3], […

selenium常见等待机制及其特点和使用方法

目录 1、强制等待 2、隐式等待 3、显式等待 1、强制等待 强制等待是在程序中直接调用Thread.sleep(timeout) ,来完成的&#xff0c;该用法的优点是使用起来方便&#xff0c;语法也比较简单&#xff0c;缺点就是需要强制等待固定的时间&#xff0c;可能会造成测试的时间过…

APP外包开发的android开发模式

开发 Android 应用有多种方法&#xff0c;每种方法都有其优势和适用场景。综合考虑各自的特点&#xff0c;你可以根据项目的需求和团队的技能选择最合适的开发方法。今天和大家分享几种常见的开发方法以及它们之间的对比&#xff0c;希望对大家有所帮助。北京木奇移动技术有限公…

探索泛型与数据结构:解锁高效编程之道

文章目录 引言第一部分&#xff1a;了解泛型1.1 为什么使用泛型1.2 使用泛型的好处 第二部分&#xff1a;泛型的使用场景2.1 类的泛型2.2 方法的泛型2.3 接口的泛型 第三部分&#xff1a;泛型通配符3.1 通配符3.2 通配符的受限泛型 第四部分&#xff1a;数据结构和泛型的应用4.…

边缘计算框架 Baetyl v2.4.3 正式发布

导读Baetyl v2.4.3 版本已经发布&#xff0c;对 v2.3.0 版本的部分功能进行了升级优化。公告称&#xff0c;这些新功能继续遵循云原生理念&#xff0c;构建了一个开放、安全、可扩展、可控制的智能边缘计算平台。 Baetyl 项目由百度发起&#xff0c;基于百度天工 AIoT 智能边缘…

【LeetCode】数据结构题解(9)[复制带随机指针的链表]

复制带随机指针的链表 &#x1f609; 1.题目来源&#x1f440;2.题目描述&#x1f914;3.解题思路&#x1f973;4.代码展示 所属专栏&#xff1a;玩转数据结构题型❤️ &#x1f680; >博主首页&#xff1a;初阳785❤️ &#x1f680; >代码托管&#xff1a;chuyang785❤…

使用Pytest集成Allure生成漂亮的图形测试报告

目录 前言 依赖包安装 Pytest Allure Pytest Adaptor 改造基于Pytest的测试用例 生成测试报告 运行测试 生成测试报告 打开测试报告 资料获取方法 前言 之前写过一篇生成测试报告的博客&#xff0c;但是其实Allure首先是一个可以独立运行的测试报告生成框架&#xff…

(杭电多校)2023“钉耙编程”中国大学生算法设计超级联赛(6)

1001 Count 当k在区间(1n)/2的左边时,如图,[1,k]和[n-k1,n]完全相同,所以就m^(n-k) 当k在区间(1n)/2的右边时,如图,[1,n-k1]和[k,n]完全相同,所以也是m^(n-k) 别忘了特判,当k等于n时,n-k为0,然后a1a1,a2a2,..anan,所以没什么限制,那么就是m^n AC代码&#xff1a; #includ…

Vue3 大屏数字滚动效果

父组件&#xff1a; <template> <div class"homePage"> <NumRoll v-for"(v, i) in numberList" :key"i" :number"v"></NumRoll> </div> </template> <script setup> import { onMounted, r…

【Vue3】动态组件

动态组件的基本使用 动态组件&#xff08;Dynamic Components&#xff09;是一种在 Vue 中根据条件或用户输入来动态渲染不同组件的技术。 在 Vue 中使用动态组件&#xff0c;可以使用 元素&#xff0c;并通过 is 特性绑定一个组件的名称或组件对象。通过在父组件中改变 is 特…

C++ 虚析构函数

在C中&#xff0c;不能声明虚构造函数&#xff0c;但是可以声明虚析构函数。 析构函数没有类型&#xff0c;也没有参数&#xff0c;和普通成员函数相比&#xff0c;虚析构函数情况很简单。 虚析构函数的声明语法&#xff1a; virtual ~类名();如果一个类的析构函数是虚函数&…

Python自动化测试用例:如何优雅的完成Json格式数据断言

目录 前言 直接使用 优化 封装 小结 进阶 总结 资料获取方法 前言 记录Json断言在工作中的应用进阶。 直接使用 很早以前写过一篇博客&#xff0c;记录当时获取一个多级json中指定key的数据&#xff1a; #! /usr/bin/python # coding:utf-8 """ aut…

森海塞尔为 CUPRA 首款纯电轿跑 SUV – CUPRA Tavascan 注入音频魅力

森海塞尔为 CUPRA 首款纯电轿跑 SUV – CUPRA Tavascan 注入音频魅力 音频专家森海塞尔携手富有挑战精神的 CUPRA&#xff0c;雕琢时代新贵车型&#xff0c;打造畅快尽兴的驾驶体验 全球知名音频专家森海塞尔与以颠覆传统、充满激情、不甘现状而闻名的汽车品牌 CUPRA 展开合作…

awk案例练习

目录 一、awk练习 1.1筛选ip地址 1.2字段去重 1.3次数统计 1.4统计TCP连接状态 1.5处理字段缺失的数据 1.6筛选给定时间范围内的日志 一、awk练习 1.1筛选ip地址 ifconfig命令查看IP 利用awk进行筛选 ifconfig | awk BEGIN{RS""}NR2{print $6} RS指定输入记…

找工作的才是大爷?面试了一个工作4年的测试工程师,一问连自动化基础都不清楚,还反过来怼我....

我们公司也开始大量招人了&#xff0c;我这次是公司招聘的面试官之一&#xff0c;主要负责一些技术上的考核&#xff0c;这段时间还真让我碰到了不少奇葩求职者 昨天公司的HR小席刚跟我吐槽&#xff1a;这几个星期没有哪天不加班的&#xff01;各种招聘网站上的消息源源不断&a…

干货分享 | TSMaster图形模块功能详解(三)—— 以CAN信号为例

TSMaster图形模块功能详解第二章节中&#xff0c;我们主要分享了显示模式、图形设置、信号跳转与波形窗口移动、信号波形缩放4大模块的操作教程。 本章节在上一篇基础上&#xff0c;继续介绍TSMaster图形模块功能第11-15模块的教程。 本文目录&#xff1a; 11、适配 11.1 水平…