广告
返回顶部
首页 > 资讯 > 前端开发 > JavaScript >JavaScript defineProperty如何实现属性劫持
  • 504
分享到

JavaScript defineProperty如何实现属性劫持

2024-04-02 19:04:59 504人浏览 独家记忆
摘要

目录前言描述符 细说get 和 set劫持对象的某个属性 劫持对象的所有属性 劫持对象的所有属性 - 包括对象类型的属性值 defineProperty的缺陷defineProper

前言

defineProperty是Vue实现数据劫持的核心,本文一点点的说明defineProperty怎么实现属性劫持的。

其实我们一般的操作对象属性的方式,增加或者修改属性,均可以使用Object.defineProperty。


let obj = {};
// 寻常操作:增加/修改 新属性
obj.a = 1;
// 等同于:
Object.defineProperty(o, "a", {
  value: 1,
  writable: true,
  configurable: true,
  enumerable: true
});

当然寻常的例子,我们是不会这么玩的,太啰嗦了。

但defineProperty可以更精确地添加或修改对象的属性。

描述符

先说个专有名词:描述符。

其实就是defineProperty的第三个参数,是个对象。这个对象的有以下属性:

  • configurable 属性:能不能修改描述符,就是能不能再次修改描述符的其他属性
  • enumerable 属性:能不能枚举该属性,就是 a 属性能不能被 for 到
  • writable 属性:能不能修改属性值,就是能不能这样修改obj.a = 1
  • value 属性:该属性的值
  • get 属性:是个函数,当访问该属性的时候,函数自动调用,函数返回值就是该属性的值
  • set 属性:是个函数,当修改该属性的时候,函数自动调用,函数有且只有一个参数,赋值的新值

注意!!!

  • 描述符里的value属性 writable属性 与 get属性 set属性是互斥的关系,只能存在一个
  • 另外的属性默认值都是false,不想false的话,记得配置哈,不细说(主要我也不怎么用)。

细说get 和 set

  • get 属性:是个函数,当访问该属性的时候,函数自动调用,函数返回值就是该属性的值
  • set 属性:是个函数,当修改该属性的时候,函数自动调用,函数有且只有一个参数,赋值的新值

默念三遍,背诵。

写个get 和 set 的例子辅助理解。

这个例子必须掌握,弄懂之后基本就掌握了数据劫持的精髓了


let obj = {};

let value = 1;
Object.defineProperty(obj, "b", {
  get() {
    console.log("读取b属性", value);
    return value;
  },
  set(newValue) {
    console.log("设置b属性", newValue);
    value = newValue;
  }
});
// 触发get函数,get的返回值就是属性值
// 1
console.log(obj.b);
// 触发set函数,value的值变成了2,注意!!!,此时内存里,属性值并没有改变
obj.b = 2;
// 但是,想要读取属性值的时候,就必然会触发get函数,属性值也自然就改变了,这个思想真的很赞
console.log(obj.b);

这里有个坑:get里是不能有读取的操作,不然一直死循环,所以使用到get set的地方,总需要借助一个变量

所以,这里,变量value的值就是属性的值,如果想要修改属性,修改 value 的值即可。

这个例子弄懂了,get,set 的精髓,我觉得也就差不多了。

劫持对象的某个属性

有了刚刚例子的基础,试着写写劫持对象的任意一个属性。


function observeKey(obj, key) {
  let value = obj[key];
  Object.defineProperty(obj, key, {
    get() {
      console.log("读取属性", value);
      return value;
    },
    set(newValue) {
      console.log("设置属性", newValue);
      value = newValue;
    }
  });
}
let obj = { a: 1 };
observeKey(obj, "a");
// 读取a,触发get函数
console.log(obj.a);
// 设置a,触发set函数
obj.a = 1;

劫持对象的所有属性

再试试劫持对象的所有属性

其实就是遍历:


function observeObj(obj) {
  for (let key in obj) {
    // 直接使用 obj.hasOwnProperty会提示不规范
    if (Object.prototype.hasOwnProperty.call(obj, key)) {
      observeKey(obj, key);
    }
  }
  return obj;
}
function observeKey(obj, key) {
  let value = obj[key];
  Object.defineProperty(obj, key, {
    get() {
      console.log("读取属性", value);
      return value;
    },
    set(newValue) {
      console.log("设置属性", newValue);
      value = newValue;
    }
  });
}

let obj = { a: 1, b: 2 };
observeObj(obj);
console.log(obj);
// 读取a,触发get函数
console.log(obj.a);
// 设置a,触发set函数
obj.a = 1;

劫持对象的所有属性 - 包括对象类型的属性值

上面的有个缺陷,就是当属性值也是对象的时候,不能劫持属性值,如{a:1,c:{b:1}}

简单,递归,补上就行。


function observeObj(obj) {
  // 加上参数限制,必须是对象才有劫持,也是递归的终止条件
  if (typeof obj !== "object" || obj == null) {
    return;
  }
  for (let key in obj) {
    // 直接使用 obj.hasOwnProperty会提示不规范
    if (Object.prototype.hasOwnProperty.call(obj, key)) {
      observeKey(obj, key);
      // 这里劫持该属性的属性值,如果不是对象直接返回,不影响
      observeObj(obj[key]);
    }
  }
  return obj;
}
function observeKey(obj, key) {
  let value = obj[key];
  Object.defineProperty(obj, key, {
    get() {
      console.log("读取属性", value);
      return value;
    },
    set(newValue) {
      console.log("设置属性", newValue);
      value = newValue;
    }
  });
}

let obj = { a: 1, b: 2, c: { name: "c" } };
observeObj(obj);
console.log(obj);
// 读取a,触发get函数
console.log(obj.a);
// 设置a,触发set函数
obj.a = 1;
// 触发set函数
obj.c.name = "d";

注意,observeObj这个函数,不能劫持对象的新增属性,只能劫持对象已有的属性。

defineProperty的缺陷

  • 不能监测对象增加属性
  • 不能监测对象删除属性
  • 不能劫持数组的修改

当然数组的修改可以通过别的方式监测到的,其是通过劫持改变数组方法实现的。

以上缺陷,也是vue里面为啥有$set/$delete以及对数组只能使用特定方法才能检测到。


let obj = { a: 1, b: [1, 2] };
observeObj(obj);
// 新增属性
obj.c = 3;
// 不会触发get函数
console.log(obj.c);
// 不会触发set函数
obj.b.push(3);

defineProperty还可以挂载属性

其实就是访问options.data.name 可以简写成 options.name,专业话术,将data上的属性挂载到options上

相当于,用defineProperty,在options上增加新属性:


// 先挂载单个属性
// options.data相当于source options相当于target
function proxyKey(target, source, key) {
  Object.defineProperty(target, key, {
    // 这里的source[key]相当于变量value,所以说最简单的那个例子是核心
    get() {
      return source[key];
    },
    set(newValue) {
      if (newValue === source[key]) {
        return;
      }
      source[key] = newValue;
    }
  });
}
// 遍历属性,挂载下
function proxyObj(target, source) {
  for (let key in source) {
    // 直接使用 obj.hasOwnProperty会提示不规范
    if (Object.prototype.hasOwnProperty.call(source, key)) {
      proxyKey(target, source, key);
    }
  }
}
let options = {
  data: { name: 1 }
};
proxyObj(options, options.data);
// 1
console.log(options.name);

话说,vue的属性劫持和挂载属性,核心原理差不多就是上面这些。

defineProperty还能写日志

比如 obj 有个属性,此属性值经常变化,想要记录其所有变化的值,以此可以形成日志。


let obj = { a: 1 };

let log = [obj.a];

let value = obj.a;
Object.defineProperty(obj, "a", {
  get() {
    return value;
  },
  set(newValue) {
    if (newValue === value) {
      return;
    }
    value = newValue;
    log.push(newValue);
  }
});

obj.a = 2;
obj.a = 3;
obj.a = 4;
// [1,2,3,4]
console.log(log);

通用的可以抽离出一个类,专门记录某个值的变化


class ArcHiver {
  constructor() {
    let value = null;
    this.archive = [];
    Object.defineProperty(this, "a", {
      get() {
        return value;
      },
      set(newValue) {
        if (newValue === value) {
          return;
        }
        value = newValue;
        this.archive.push(newValue);
      }
    });
  }
}
let archiver = new Archiver();
archiver.a = 1;
archiver.a = 2;
// [1,2]
console.log(archiver.archive);

引用

MDN的defineProperty

总结

到此这篇关于javascript defineProperty如何实现属性劫持的文章就介绍到这了,更多相关defineProperty属性劫持内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: JavaScript defineProperty如何实现属性劫持

本文链接: https://www.lsjlt.com/news/130706.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • JavaScript defineProperty如何实现属性劫持
    目录前言描述符 细说get 和 set劫持对象的某个属性 劫持对象的所有属性 劫持对象的所有属性 - 包括对象类型的属性值 defineProperty的缺陷defineProper...
    99+
    2022-11-12
  • JavaScript defineProperty怎么实现属性劫持
    本篇内容主要讲解“JavaScript defineProperty怎么实现属性劫持”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“JavaScript defineProperty怎么实现属性劫...
    99+
    2023-06-20
  • Ajax如何实现网站劫持的检测方法
    这篇文章将为大家详细讲解有关Ajax如何实现网站劫持的检测方法,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。https可以彻底解决劫持的问题。但是一般虚拟主机都不支持 https,难道http只能任流氓们...
    99+
    2023-06-08
  • JavaScript如何实现动态属性名
    小编给大家分享一下JavaScript如何实现动态属性名,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!动态属性名我一直以为必须先声明一个对象,然后才能分配动态属性...
    99+
    2023-06-27
  • 如何实现javascript操作input标签属性checkbox全选
    这篇文章主要为大家展示了“如何实现javascript操作input标签属性checkbox全选”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“如何实现javas...
    99+
    2022-10-19
  • css3中transform属性如何实现
    这篇文章主要介绍css3中transform属性如何实现,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!   1transform属性   在CSS3中,可以利用transform...
    99+
    2022-10-19
  • vue如何实现计算属性
    本文小编为大家详细介绍“vue如何实现计算属性”,内容详细,步骤清晰,细节处理妥当,希望这篇“vue如何实现计算属性”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。什么是计算属性模板内的表达式非常便利,但是设计它们...
    99+
    2023-07-04
  • css中如何实现动画属性
    这篇文章主要介绍了css中如何实现动画属性,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。   一、css动画(animation)属性可以...
    99+
    2022-10-19
  • css如何实现简单属性选择
    这篇文章给大家分享的是有关css如何实现简单属性选择的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。 如果希望选择有某个属性的元素,而不论属性值是什么,可以使用简单属性选择器。 ...
    99+
    2022-10-19
  • Vue3计算属性是如何实现的
    目录计算属性使用微任务优化调度器前言: 本篇内容基于Vue3响应式对象是如何实现的(2)实现。 计算属性 Vue3的官方文档中,对于计算属性有这样的描述: 对于任何包含响应式数据的复...
    99+
    2022-11-13
  • vue如何实现属性事件传递
    这篇文章给大家分享的是有关vue如何实现属性事件传递的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。属性事件传递写过高阶组件的童鞋可能都会碰到过将加工过的属性向下传递的情况,如果碰...
    99+
    2022-10-19
  • 使用Spring如何实现注入属性
    使用Spring如何实现注入属性?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。在spring框架中,属性的注入我们有多种方式,我们可以通过构造方法注入,可以通过...
    99+
    2023-05-31
    spring 属性注入
  • JavaScript如何实现基础类型、对象一样有属性和方法
    这篇文章主要介绍“JavaScript如何实现基础类型、对象一样有属性和方法”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“JavaScript如何实现基础类型、对象...
    99+
    2022-10-19
  • C#中如何使用属性和自动实现属性简化代码
    C#中如何使用属性和自动实现属性简化代码,需要具体代码示例在C#编程中,使用属性和自动实现属性可以帮助我们简化代码,提高代码的可读性和可维护性。属性允许我们通过封装字段来访问和更改对象的状态,而自动实现属性则进一步简化了属性的创建过程。属性...
    99+
    2023-10-22
    属性(属性) 自动实现属性(自动属性) 代码简化(简化代码)
  • Mybatis如何实现关联属性懒加载
    Mybatis 关联属性懒加载 延迟加载配置 mybatis默认没有开启延迟加载,需要在config.xml中setting配置。 lazyLoadingEnabled:true使用...
    99+
    2022-11-12
  • 如何使用JavaScript实现select所支持的功能
    小编给大家分享一下如何使用JavaScript实现select所支持的功能,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧...
    99+
    2022-10-19
  • 如何用Table-cell属性让div实现定位
    本篇内容介绍了“如何用Table-cell属性让div实现定位”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成...
    99+
    2022-10-19
  • PHP中Grafika如何实现图像属性处理
    这篇文章主要为大家展示了“PHP中Grafika如何实现图像属性处理”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“PHP中Grafika如何实现图像属性处理”这篇文章吧。1、图片格式化为二进制格...
    99+
    2023-06-17
  • 电脑出现windows网页对象不支持此属性或方法如何解决
    这篇“电脑出现windows网页对象不支持此属性或方法如何解决”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“电脑出现wind...
    99+
    2023-06-27
  • Angular2如何实现自定义双向绑定属性
    这篇文章将为大家详细讲解有关Angular2如何实现自定义双向绑定属性,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。整理文档,搜刮出一个Angular 2实现自定义 双向...
    99+
    2022-10-19
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作