JS笔试手撕题

数据劫持

Vue2的Object.defineProperty()

Vue2的响应式是通过Object.defineProperty()拦截数据,将数据转换成getter/setter的形式,在访问数据的时候调用getter函数,在修改数据的时候调用setter函数。然后利用发布-订阅模式,在数据变动时触发依赖,也即发布更新给订阅者,订阅者收到消息后进行相应的处理

描述符分为数据描述符和存取描述符,只能是其中之一,不可两者同时存在
在这里插入图片描述
configurable:表示能否通过delete删除属性从而重新定义属性,能否修改属性的特性,或者能否把属性修改为访问器属性,默认值为false。
enumerable:表示能否通过for in循环访问属性,默认值为false
//数据描述符
1.writable:表示能否修改属性的值。默认值为false。
2.value:包含这个属性的数据值。默认值为undefined。
//存取描述符
1.get:在读取属性时调用的函数,默认值是undefined
2.set:在写入属性的时候调用的函数,默认值是undefined

  <script>function defineReactive(data) {if (!data || Object.prototype.toString.call(data) !== '[object Object]')returnfor (let key in data) {let val = data[key]Object.defineProperty(data, key, {configurable: true,//可配置enumerable: true,//可枚举set: function (newval) {console.log('调用set');if (newval === val) {return ;}console.log('属性值发生变化');val = newval},get() {console.log('调用get');return val},})if (typeof val === 'object') {defineReactive(val)}}}const data = {name: 'better',firends: ['1', '2']}defineReactive(data)console.log('data.friends[0]:', data.firends[0]);console.log('data.name:', data.name);data.name = 'lihua'console.log('更改后的data.name:', data.name);data.friends[0] = '100'// 在JavaScript中,数组是一种特殊的对象,因此直接对数组进行defineProperty并不能监听数组元素的变化。// 当我们尝试修改数组元素时,比如data.friends[0] = 100,实际上并不会触发set函数,也就无法实现数组元素的响应式更新。console.log('更改后的data.friends[0]:', data.firends[0]);</script>

在这里插入图片描述
弊端:
1.无法劫持对象的添加和删除操作,Vue2的解决方法(新增set和delete)
2.无法劫持到数组的api(push、pop…),Vue2的解决方法是重写数组的api
3.存在深层嵌套关系,通过无脑递归,但也会劫持到项目始终都不会用到的对象,造成性能损耗

Vue3的proxy(解决vue2响应式的弊端)

Proxy(代理):用于创建一个对象的代理,从而实现基本操作的拦截和自定义(如属性查找、赋值、枚举、函数调用等)。
Reflect(反射):对源对象的属性进行操作

Vue3基于ES6新增的Proxy对象实现数据代理以及通过Reflect对源数据进行操作,它解决了Vue2中无法追踪数据新增或删除属性的问题。另外,Proxy可以直接监听数组,无需像Vue2响应式那样需要重写数组方法进行拦截

proxy详解

const p = new Proxy(target, handler)
//target:要使用 Proxy 包装的目标对象(可以是任何类型的对象,包括原生数组,函数,甚至另一个代理)。
//handler:一个通常以函数作为属性的对象,各属性中的函数分别定义了在执行各种操作时代理 p 的行为。
 <script>function isObject(obj) {if (typeof obj !== "object" || !obj) {return false;}return true;}// vue3使用reactive定义响应式对象时返回的就是reactive的实例function reactive(obj) {// 对整个对象进行劫持return new Proxy(obj, {get(target, key) {console.log('get:', key);// proxy想要监听深层对象也需要进行递归处理,不过这里用懒加载,用不到的属性不用进行深层监听let res = Reflect.get(target, key)return isObject(res) ? reactive(res) : res;},set(target, key, value) {console.log('set:', key, value);return Reflect.set(target, key, value)},deleteProperty(target, key) {console.log('delete', key);return Reflect.deleteProperty(target, key)}})}let person = reactive({ name: 'zs', age: 18, job: { code: "前端" } })console.log('person.name:', person.name);// 直接对proxy实例操作person.name = '小张'console.log('------------');person.gender = '男'delete person.ageconsole.log('------------');console.log('测试对数组的监听');//数组的push方法,作为push的属性理应也该被劫持到let arr = reactive([1, 2, 3, 4, 5])arr.push(11)console.log('--------------------');console.log('对深层对象的测试');console.log('person.job.code:', person.job.code);person.job.code = "php"</script>

在这里插入图片描述

总结
优点

1.Proxy直接劫持整个对象,并返回一个新对象,我们可以只操作新的对象达到响应式的目的
2.Proxy可以直接监听数组的变化(push,shift,splice…)
3.Proxy有多达13中拦截方法,不限于apply、ownKeys、deleteProperty、has等等,这是Object.defineProperty所不具备的
4.Proxy懒加载,解决递归造成的性能问题

缺点

Proxy不兼容IE

闭包

 <script>function fn1() {debugger;let num = 999;function innerFn1() {debugger;console.log(num);}innerFn1()}fn1()</script>

在这里插入图片描述
函数依赖外部变量就形成闭包
在这里插入图片描述

闭包的应用场景

函数内部return出去一个函数,就是结合了闭包的应用场景之一,目的是为了让那个闭包持久化

    <script>function fn1() {debugger;let num = 999;function innerFn1() {debugger;console.log(num);}return innerFn1}let closure1=fn1();//闭包 num:999// 两个不同的闭包,因此闭包可以解决全局变量命名冲突问题let closure2=fn1();//闭包 num:999</script>
<script>// 面试题function fn3(a, b) {console.log('第二个参数的值是:', b);return {innerFn: function (c) {return fn3(c, a);}}}// 链式调用fn3(1).innerFn(2).innerFn(3).innerFn(4)// 1.外部调用 传递a:1 返回一个对象 形成闭包 (a:1)=>输出undefined// 2.传递c:2 执行函数,并返回一个对象  fn3(c,a) (c:2,a:1)=>输出1// 3.外部a:2,b:1,返回一个对象(a:2)// 4.参数c:3,执行fn(3,2)   外部:(a:3,b:2)=>输出2// 5.闭包(a:3)</script>

在这里插入图片描述

        // 面试题function fn(a, b) {debuggerconsole.log('第二个参数的值是:', b);return {//对象中的函数形成一个闭包fn: function (c) {debuggerreturn fn(c, a)}}}let obj = fn(1)// 一个新对象有一个闭包=>1个// 访问的都是同一个闭包obj.fn(2)obj.fn(3)obj.fn(4)

在这里插入图片描述

 <script>// 面试题function fn(a, b) {debuggerconsole.log('第二个参数的值是:', b);return {//对象中的函数形成一个闭包fn: function (c) {debuggerreturn fn(c, a)}}}// let obj = fn(1)// // 一个新对象有一个闭包=>1个// // 访问的都是同一个闭包// obj.fn(2)// obj.fn(3)// obj.fn(4)console.log('----------------');let a = fn(1).fn(2)a.fn(3).fn(4)a.fn(5).fn(6)</script>

在这里插入图片描述

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

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

相关文章

基于PSO粒子群优化的配电网可靠性指标matlab仿真

目录 1.程序功能描述 2.测试软件版本以及运行结果展示 3.核心程序 4.本算法原理 4.1 PSO算法应用于配电网优化的基本原理 5.完整程序 1.程序功能描述 基于PSO粒子群优化的配电网可靠性指标matlab仿真&#xff0c;指标包括saifi, saidi, caidi, aens四个。 2.测试软件版本…

深度学习--DCGAN

代码之后的注释和GAN的一样&#xff0c;大家如果已经掌握GAN&#xff0c;可以忽略掉哦&#xff01;&#xff01;&#xff01; 在学习DCGAN之前&#xff0c;我们要先掌握GAN&#xff0c;深度学习--生成对抗网络GAN-CSDN博客 这篇博客讲的就是GAN的相关知识&#xff0c;还是很详…

POST请求

1、代码 import urllib.request import urllib.parse# 指定 URL url https://fanyi.baidu.com/sug# POST 请求携带的参数进行处理流程&#xff1a; # 1. 将 POST 请求参数封装到字典 data {kw: 西瓜 }# 2. 使用 parse 模块中的 urlencode 进行编码处理 data urllib.parse.u…

【JAVA进阶篇教学】第十篇:Java中线程安全、锁讲解

博主打算从0-1讲解下java进阶篇教学&#xff0c;今天教学第十篇&#xff1a;Java中线程安全、锁讲解。 当涉及到多线程编程时&#xff0c;保证线程安全是至关重要的。线程安全意味着在多个线程访问共享资源时&#xff0c;不会发生数据错乱或不一致的情况。为了实现线程安全&am…

JavaScript异步编程——05-回调函数

我们在前面的文章《JavaScript 基础&#xff1a;异步编程/单线程和异步》中讲过&#xff0c;Javascript 是⼀⻔单线程语⾔。早期我们解决异步场景时&#xff0c;⼤部分情况都是通过回调函数来进⾏。 &#xff08;如果你还不了解单线程和异步的概念&#xff0c;可以先去回顾上一…

【Redis7】10大数据类型之Zset类型

文章目录 1.Zset类型2.常用命令3.示例3.1 ZADD,ZRANGE和ZREVRANGE3.2 ZSCORE,ZCARD和ZREM3.3 ZRANGEBYSCORE和ZCOUNT3.4 ZRANK和ZREVRANK3.5 Redis7新命令ZMPOP 1.Zset类型 Redis的Zset&#xff08;Sorted Set&#xff0c;有序集合&#xff09;是一种特殊的数据结构&#xff0…

Encoder——Decoder工作原理与代码支撑

神经网络算法 &#xff1a;一文搞懂 Encoder-Decoder&#xff08;编码器-解码器&#xff09;_有编码器和解码器的神经网络-CSDN博客这篇文章写的不错&#xff0c;从定性的角度解释了一下&#xff0c;什么是编码器与解码器&#xff0c;我再学习笔记补充的时候&#xff0c;讲一下…

TMS320F28335学习笔记-时钟系统

第一次使用38225使用了普中的clocksystem例程进行编译&#xff0c;总是编译失败。 问题一&#xff1a;提示找不到文件 因为工程的头文件路径没有包含&#xff0c;下图的路径需要添加自己电脑的路径。 问题二 找不到库文件 例程种的header文件夹和common文件夹不知道从何而来…

【Alluxio】文件系统锁模型之InodeLockList

InodeLockList接口,表示在inode tree里一个加了锁的路径。 沿着path,inodes和edges都被加锁了。path可能从edge或inode任意一个开始。 锁列表总是包含了一定数量的读锁(0个或多个),随后跟随着一些数量的写锁(0个或多个)。 举个例子: 对 /a/b/c/d 进行加锁,c->d这…

【深度学习】网络安全,SQL注入识别,SQL注入检测,基于深度学习的sql注入语句识别,数据集,代码

文章目录 一、 什么是sql注入二、 sql注入的例子三、 深度学习模型3.1. SQL注入识别任务3.2. 使用全连接神经网络来做分类3.3. 使用bert来做sql语句分类 四、 深度学习模型的算法推理和部署五、代码获取 一、 什么是sql注入 SQL注入是一种常见的网络安全漏洞&#xff0c;它允许…

【进程间通信】共享内存

文章目录 共享内存常用的接口指令利用命名管道实现同步机制总结 System V的IPC资源的生命周期都是随内核的。 共享内存 共享内存也是为了进程间进行通信的&#xff0c;因为进程间具有独立性&#xff0c;通信的本质是两个不同的进程看到同一份公共资源&#xff0c;所以共享内存…

Java 11 到 Java 8 的兼容性转换

Java 11 到 Java 8 的兼容性转换 欲倚绿窗伴卿卿&#xff0c;颇悔今生误道行。有心持钵丛林去&#xff0c;又负美人一片情。 静坐修观法眼开&#xff0c;祈求三宝降灵台&#xff0c;观中诸圣何曾见&#xff1f;不请情人却自来。 入山投谒得道僧&#xff0c;求教上师说因明。争奈…

WordPress MasterStudy LMS插件 SQL注入漏洞复现(CVE-2024-1512)

0x01 产品简介 WordPress和WordPress plugin都是WordPress基金会的产品。WordPress是一套使用PHP语言开发的博客平台。该平台支持在PHP和MySQL的服务器上架设个人博客网站。WordPress plugin是一个应用插件。 0x02 漏洞概述 WordPress Plugin MasterStudy LMS 3.2.5 版本及之…

java项目之在线课程管理系统源码(springboot+vue+mysql)

风定落花生&#xff0c;歌声逐流水&#xff0c;大家好我是风歌&#xff0c;混迹在java圈的辛苦码农。今天要和大家聊的是一款基于springboot的在线课程管理系统。项目源码以及部署相关请联系风歌&#xff0c;文末附上联系信息 。 项目简介&#xff1a; 在线课程管理系统的主要…

Nginx配置/.well-known/pki-validation/

当你需要在Nginx上配置.well-known/pki-validation/时&#xff0c;这通常是为了支持SSL证书的自动续订或其他验证目的。以下是配置步骤&#xff1a; 创建目录结构&#xff1a; 在你的网站根目录下创建一个名为.well-known的目录&#xff08;SSL证书申请之如何创建/.well-known/…

Linux环境Redis部署

Redis部署 Redis是一个高性能的开源键值存储系统&#xff0c;它主要基于内存操作&#xff0c;但也支持数据的持久化。与其他数据库相比&#xff0c;Redis的主要优势在于它的高性能、丰富的数据结构和原生的持久化能力。Redis不仅提供了类似的功能&#xff0c;还增加了持久化和…

[初阶数据结构】单链表

前言 &#x1f4da;作者简介&#xff1a;爱编程的小马&#xff0c;正在学习C/C&#xff0c;Linux及MySQL。 &#x1f4da;本文收录于初阶数据结构系列&#xff0c;本专栏主要是针对时间、空间复杂度&#xff0c;顺序表和链表、栈和队列、二叉树以及各类排序算法&#xff0c;持…

如何使用client-go构建pod web shell

代码示例及原理 原理是利用websocket协议实现对pod的exec登录&#xff0c;利用client-go构造与远程apiserver的长连接&#xff0c;将对pod容器的输入和pod容器的输出重定向到我们的io方法中&#xff0c;从而实现浏览器端的虚拟终端的效果消息体结构如下 type Connection stru…

Meta更低的训练成本取得更好的性能: 多token预测(Multi-Token Prediction)

Meta提出了一种透过多token预测(Multi-token Prediction)来训练更好、更快的大型语言模型的方法。这篇论文的重点如下: 训练语言模型同时预测多个未来的token,可以提高样本效率(sample efficiency)。 在推论阶段,使用多token预测可以达到最高3倍的加速。 论文的主要贡献包括: …

ES集群数据备份与迁移

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言一、文章涉及概念讲解二、操作步骤1.创建 snapshot repository操作主机hadoop1分别操作从机hadoop2和hadoop3 2. 查看仓库信息3. 备份索引&#xff0c;生成快照…