当前位置: 首页 > news >正文

web前端面试高频考点——Vue3.x(Composition API的逻辑复用、Proxy实现响应式)

系列文章目录

内容参考链接
JavaScript 面试高频考点HTML、CSS、JavaScript、ES6、AJAX、HTTP 面试考点
Vue2.x 面试高频考点Vue2.x 面试高频考点
Vue3.x新增API生命周期,ref、toRef 和 toRefs 的理解和最佳使用方式
Vue3.x升级的重要功能emits属性、生命周期、多事件、Fragment、移出.async、异步组件写法、移出 filter、Teleport、Suspense…

文章目录

  • 系列文章目录
    • 一、Composition API 如何实现逻辑复用
    • 二、Vue3 如何实现响应式
      • 1、Object.defineProperty 的缺点
      • 2、Proxy 实现响应式
      • 3、Reflect 作用
      • 4、Proxy 实现响应式


一、Composition API 如何实现逻辑复用

  • 抽离逻辑代码到一个函数
  • 函数命名约定为 useXxx 格式(React Hooks 也是)
  • 在 setup 中引用 useXxx 函数

useMousePosition.js 文件

  • 鼠标移动事件,显示鼠标的位置
  • 写在 js 文件中,可供逻辑复用
import { ref, onMounted, onUnmounted} from 'vue'

function useMousePosition() {
    // 初始化坐标
    const x = ref(0)
    const y = ref(0)

    // 更新坐标
    function update(e) {
        x.value = e.pageX
        y.value = e.pageY
    }

    // 挂载:添加鼠标移动事件
    onMounted(() => {
        console.log('useMousePosition mounted');
        window.addEventListener('mousemove', update)
    })

    // 销毁:删除鼠标移动事件
    onUnmounted(() => {
        console.log('useMousePosition unMounted');
        window.removeEventListener('mousemove', update)
    })

    return {
        x,
        y
    }
}

export default useMousePosition

App.vue 父组件

  • 点击按钮,进行组件的创建 / 销毁
<template>
  <MousePosition v-if="flag" />
  <button @click="changeFlagHandler">change flag</button>
</template>

<script>
import MousePosition from "./components/index.vue";
export default {
  data() {
    return {
      flag: true,
    };
  },
  methods: {
    // 实现组件的创建/销毁
    changeFlagHandler() {
      this.flag = !this.flag;
    },
  },
  components: { MousePosition },
};
</script>

index.vue 子组件

  • 解构出函数中定义的 x 和 y
<template>
  <p>mouse position {{ x }} {{ y }}</p>
</template>

<script>
import useMousePosition from "./useMousePosition";

export default {
  name: "MousePosition",
  setup() {
    // 解构 x 和 y
    const { x, y } = useMousePosition();

    return {
      x,
      y,
    };
  },
};
</script>

CompositionAPI复用

二、Vue3 如何实现响应式

1、Object.defineProperty 的缺点

  • 深度监听需要一次性递归(层级很深的话会影响性能)
  • 无法监听新增属性 / 删除属性(Vue.set Vue.delete)
  • 无法原生监听数组,需要特殊处理

2、Proxy 实现响应式

  • target:就是定义的对象 data
  • key:获取的键
  • val:获取的值
  • receiver:是 proxyData

示例:对象通过 Proxy 实现响应式测试

const data = {
    name: '杂货铺',
    age: 20
}

const proxyData = new Proxy(data, {
	// 监听获取
    get(target, key, receiver) {
        const result = Reflect.get(target, key, receiver)
        console.log('get', key);
        return result // 返回结果
    },
    
    // 监听设置
    set(target, key, val, receiver) {
        const result = Reflect.set(target, key, val, receiver)
        console.log('set', key, val)
        console.log('result', result); // true
        return result // 是否设置成功
    },
    
    // 监听删除
    deleteProperty(target, key) {
        const result = Reflect.deleteProperty(target, key)
        console.log('delete property', key);
        console.log('result', result); // true
        return result // 是否删除成功
    }
})

在这里插入图片描述

示例:数组通过 Proxy 实现响应式测试

const data = ['a', 'b', 'c']

const proxyData = new Proxy(data, {
    get(target, key, receiver) {
        // 只处理本身(非原型的)属性
        const ownKeys = Reflect.ownKeys(target) // 获取对象的键
        if (ownKeys.includes(key)) {
            console.log('get', key);  // 监听
        }
        const result = Reflect.get(target, key, receiver)
        console.log('get', key);
        return result // 返回结果
    },
    set(target, key, val, receiver) {
        // 重复的数据,不处理
        const oldVal = target[key]
        if(val === oldVal) {
            return true
        }
        const result = Reflect.set(target, key, val, receiver)
        console.log('set', key, val)
        console.log('result', result); // true
        return result // 是否设置成功
    },
    deleteProperty(target, key) {
        const result = Reflect.deleteProperty(target, key)
        console.log('delete property', key);
        console.log('result', result); // true
        return result // 是否删除成功
    }
})

在这里插入图片描述

3、Reflect 作用

  • 和 Proxy 能力一一对应
  • 规范化、标准化、函数式
  • 替代掉 Object 上的工具函数
    在这里插入图片描述

4、Proxy 实现响应式

  • 深度监听,性能更好(用到的时候再监听)
  • 可监听 新增 / 删除 属性
  • 可监听数组变化
  • Proxy 能规避 Object.defineProperty 的问题
  • Proxy 无法兼容所有浏览器,无法 polyfill(用于实现浏览器并不支持原生 API 的代码)

示例:使用 Proxy 实现响应式

  • 深度监听不是一次性监听完,而是用到的时候才监听
// 创建响应式
function reactive(target = {}) {
    if (typeof target !== 'object' || target == null) {
        // 不是对象或数组
        return target
    }

    // 代理配置
    const proxyConf = {
        get(target, key, receiver) {
            // 只处理本身(非原型的)属性
            const ownKeys = Reflect.ownKeys(target)
            if (ownKeys.includes(key)) {
                console.log('get', key); // 监听
            }
            const result = Reflect.get(target, key, receiver)

            // 深度监听
            // 性能如何提升的? 什么时候 get 到,什么时候去做响应式
            return reactive(result) // 返回结果
        },
        set(target, key, val, receiver) {
            // 重复的数据,不处理
            const oldVal = target[key]
            if (val === oldVal) {
                return true
            }

            // 监听是已有的键还是新增的键
            const ownKeys = Reflect.ownKeys(target)
            if (ownKeys.includes(key)) {
                console.log('已有的 key', key);
            } else {
                console.log('新增的 key', key);
            }

            const result = Reflect.set(target, key, val, receiver)
            console.log('set', key, val)
            return result // 是否设置成功
        },
        deleteProperty(target, key) {
            const result = Reflect.deleteProperty(target, key)
            console.log('delete property', key);
            console.log('result', result); // tru            return result // 是否删除成功
        }
    }

    // 生成代理对象
    const observed = new Proxy(target, proxyConf)
    return observed
}

// 测试数据
const data = {
    name: '杂货铺',
    age: 21,
    info: {
        city: 'beijing'
    }
}

const proxyData = reactive(data)

不积跬步无以至千里 不积小流无以成江海

点个关注不迷路,持续更新中…

相关文章:

  • go语言实现十大排序算法
  • 学习笔记(13)ES6新特性
  • vue2 项目中引入iconfont
  • 手把手教你深度学习和实战-----循环神经网络(RNN、LSTM)
  • 西宾猫耳下载工具(missevandown)
  • 【C++项目】boost搜索引擎
  • 9.5-9.9 小知识点
  • 【流行框架】SpringMVC
  • Linux命令之chage命令
  • java导出功能(多个sheet页数据导出)
  • 力扣系列题,回溯专场
  • 在windows桌面上部署网站
  • 8月更新 | Visual Studio Code Python
  • 猿创征文 |【算法面试入门必刷】动态规划-线性dp(一)
  • 一整套美团面经(给对象超用心整理的)
  • 【干货分享】SpringCloud微服务架构分布式组件如何共享session对象
  • 2017 前端面试准备 - 收藏集 - 掘金
  • Essential Studio for ASP.NET Web Forms 2017 v2,新增自定义树形网格工具栏
  • JavaScript类型识别
  • JavaSE小实践1:Java爬取斗图网站的所有表情包
  • Java读取Properties文件的六种方法
  • MaxCompute访问TableStore(OTS) 数据
  • Python进阶细节
  • SAP云平台里Global Account和Sub Account的关系
  • 关于 Cirru Editor 存储格式
  • 机器学习 vs. 深度学习
  • 技术胖1-4季视频复习— (看视频笔记)
  • 排序算法之--选择排序
  • 使用前端开发工具包WijmoJS - 创建自定义DropDownTree控件(包含源代码)
  • 栈实现走出迷宫(C++)
  • ​DB-Engines 11月数据库排名:PostgreSQL坐稳同期涨幅榜冠军宝座
  • ​LeetCode解法汇总518. 零钱兑换 II
  • # Maven错误Error executing Maven
  • #Linux(权限管理)
  • #调用传感器数据_Flink使用函数之监控传感器温度上升提醒
  • (2021|NIPS,扩散,无条件分数估计,条件分数估计)无分类器引导扩散
  • (C语言)球球大作战
  • (Java实习生)每日10道面试题打卡——JavaWeb篇
  • (Spark3.2.0)Spark SQL 初探: 使用大数据分析2000万KF数据
  • (牛客腾讯思维编程题)编码编码分组打印下标题目分析
  • (七)Java对象在Hibernate持久化层的状态
  • (七)理解angular中的module和injector,即依赖注入
  • (七)微服务分布式云架构spring cloud - common-service 项目构建过程
  • (一)C语言之入门:使用Visual Studio Community 2022运行hello world
  • .Net FrameWork总结
  • .NET 简介:跨平台、开源、高性能的开发平台
  • .net生成的类,跨工程调用显示注释
  • /etc/skel 目录作用
  • @selector(..)警告提示
  • []sim300 GPRS数据收发程序
  • [20150629]简单的加密连接.txt
  • [2669]2-2 Time类的定义
  • [BZOJ] 3262: 陌上花开
  • [bzoj1038][ZJOI2008]瞭望塔
  • [C++]C++基础知识概述