Object.defineProperty第三个参数descriptor的说明。

数据描述符和存取描述符均具有以下可选键值:

定义了 value 或 writable , 一定不能有 get 或 set, 反之亦然, 否则报错.

configurable

当且仅当该属性的 configurable 为 true 时,该属性描述符才能够被改变,同时该属性也能从对应的对象上被删除。默认为 false。

enumerable

当且仅当该属性的enumerable为true时,该属性才能够出现在对象的枚举属性中。默认为 false。

数据描述符同时具有以下可选键值:

value

该属性对应的值。可以是任何有效的 JavaScript 值(数值,对象,函数等)。默认为 undefined。

可以将属性函数化。

eg:

var student = {
age: 14,
sayHi: function (name){
console.log('hi~' + name);
}
}
Object.defineProperty(student,'age',{
configurable: true,
enumerable: true,
value(){
return 1;
},
})
student.age // ƒ value(){ return 1; }
student.age() // 1

writable

当且仅当该属性的writable为true时,value才能被赋值运算符改变。默认为 false。

存取描述符同时具有以下可选键值:

get

一个给属性提供 getter 的方法,如果没有 getter 则为 undefined。当访问该属性时,该方法会被执行,方法执行时没有参数传入,但是会传入this对象(由于继承关系,这里的this并不一定是定义该属性的对象)。

默认为 undefined。

set

一个给属性提供 setter 的方法,如果没有 setter 则为 undefined。当属性值修改时,触发执行该方法。该方法将接受唯一参数,即该属性新的参数值。

默认为 undefined。

class Observer {
constructor(data) {
this.data = data;
this.filterObj(data);
}
static isObject(obj) {
if (Object.prototype.toString.call(obj) === "[object Object]") {
return true;
}
return false; }
filterObj(data) {
if (!Observer.isObject(data)) return;
for (const key in data) {
// 过滤原型链上的属性。
if (data.hasOwnProperty(key)) {
const value = data[key];
if (Observer.isObject(data[key])) {
new Observer(data[key]);
};
this.watch(key, value);
}
}
}
watch(k, v) {
Object.defineProperty(this.data, k, {
enumerable: true,
configurable: true,
get: function () {
console.log(`${k},被访问。`)
return v;
},
set: function (newV) {
console.log(`${k},属性值发生变化。`)
console.log(`新的值为:${JSON.stringify(newV)}。`)
if (Observer.isObject(newV)) {
new Observer(newV);
}
v = newV;
},
})
}
}
let data = {
time: '2048',
user: {
name: 'naruto',
equipment: {
arms: 'kuwu',
ArmGuard: 'long',
}
},
}; const app = new Observer(data);

/*
Array.prototype重写method
getOwnPropertyDescriptor方法可以查看原生js的push等方法是否可以重写。
MDN:
Object.getOwnPropertyDescriptor(obj, prop)通过方法可查看自有属性对应的属性描述符。
(自有属性指的是直接赋予该对象的属性,不需要从原型链上进行查找的属性)
Object.getOwnPropertyDescriptor(Array.prototype, "push");
{value: ƒ push(), writable: true, enumerable: false, configurable: true}
*/
Object.defineProperty(Array.prototype, "push", {
configurable: true,
enumerable: false,
writable: true,
value: function () {
const arg = [].slice.call(arguments);
// console.log('');
const len = this.length;
for (let i = 0; i < arg.length; i++) {
this[len + i] = arg[i]
}
return this.length;
}
}); // Array.prototype.push = function () {
// ...
// } /*
MDN:
Array.isArray(Array.prototype); // true;
Array.prototype[0]; // undefined
鲜为人知的事实:Array.prototype 本身也是一个 Array。
Array拥有的静态方法:from isArray of。 Array.prototype.constructor
所有的数组实例都继承了这个属性,它的值就是 Array,表明了所有的数组都是由 Array 构造出来的。
Array.prototype.length
上面说了,因为 Array.prototype 也是个数组,所以它也有 length 属性,这个值为 0,因为它是个空数组。 look like this:
Array.prototype.first = function() {};
Array.prototype.mapping = function() {};
Array.prototype["mapping"];
*/

操作对象中的Getter

Object.defineProperty(Array.prototype,'fn',{
configurable:true,
enumerable:true,
get(){
const test = function(){};
test.f1 = ()=>{
console.log(this)
};
return test;
}
});
[1,2,3].fn.f1() //[1, 2, 3] Array.prototype.__defineGetter__('fn',function(){
//const a = {};
//const a = function(){};
a.f1 = ()=>{
console.log(this);
};
return a;
});
[1,2,3].fn.f1() //[1, 2, 3] var a = {};
a.__defineGetter__('name',function(){
return 'wfc';
});
a.name //'wfc

js 数据监听--对象的变化的更多相关文章

  1. Object.defineProperty 监听对象属性变化

    <!DOCTYPE html> <html lang="zh"> <head> <meta charset="UTF-8&quo ...

  2. vue 2.0 watch 监听对象的变化

  3. Proxy监听对象的数据变化,处理绑定数据很有用

    Proxy可以监听对象身上发生了什么事情,并在这些事情发生后执行一些相应的操作.一下子让我们对一个对象有了很强的追踪能力,同时在数据绑定方面也很有用处. }; //interceptor 拦截 var ...

  4. js动态监听dom变化

    原生js 动态监听dom变化,根据不同的类型绑定不同的处理逻辑 // Firefox和Chrome早期版本中带有前缀   var MutationObserver = window.MutationO ...

  5. vue 监听对象里的特定数据

    vue  监听对象里的特定数据变化 通常是这样写的,只能监听某一个特定数据 watch: { params: function(val) { console.log(val) this.$ajax.g ...

  6. js 实时监听input中值变化

    注意:用到了jquery需要引入jquery.min.js. 需求: 1.每个地方需要分别打分,总分为100; 2.第一个打分总分为40; 3.第二个打分总分为60. 注意:需要判断null.&quo ...

  7. js/jquery 实时监听输入框值变化的完美方案:oninput & onpropertychange

    (1)     先说jquery, 使用 jQuery 库的话,只需要同时绑定 oninput 和 onpropertychange 两个事件就可以了,示例代码: $('#username').bin ...

  8. JS 获取和监听屏幕方向变化(portrait / landscape)

    移动设备的屏幕有两个方向: landscape(横屏)和portrait(竖屏),在某些情况下需要获取设备的屏幕方向和监听屏幕方向的变化,因此可以使用Javascript提供的 MediaQueryL ...

  9. vue--》如何使用wacth监听对象的属性变化?

    在开发过程中,我们经常需要监听watch监听一个对象的变化,但是如何来实现     监听对象中属性的变化呢? 先回顾一下如何监听整个对象的变化,使用watch就行了 export default { ...

随机推荐

  1. BurpSuite系列----Extender模块(扩展器)

    一.简介 Burp在软件中提供了支持第三方拓展插件的功能,方便使用者编写自己的自定义插件或从插件商店中安装拓展插件.Burp扩展程序可以以多种方式支持自定义Burp的行为,例如:修改HTTP请求和响应 ...

  2. FPGA实战操作(2) -- PCIe总线(协议简述)

    目录 1. PCIe基础知识 2. 事务层协议 2.1 数据包结构 2.2 帧头含义详述 3. 报文举例 3.1 寄存器读报文 3.2 完成报文 4. 机制简述 4.1 Non-Posted和Post ...

  3. ADC新库

    1.单次采集模式 1.在STM32CUBMX中设置为单次采集模式 2.在C文件中用HAL_ADC_START()函数启动ADC 3.用HAL_ADC_PollForConversion()延时等待采集 ...

  4. 关于README的内容

    如何在开发项目的时候写出标准的README文件? 为什么要写这篇博客? 关于README的编写,这个也十分的重要,如果你觉得只是随便写写,那么你就错了:编写这部分也是相当的重要的.工欲善其事,必先利其 ...

  5. mono修改代码模板

    新建android application是在这里修改模板D:\prostu\Microsoft Visual Studio 10.0\Common7\IDE\ProjectTemplatesCach ...

  6. 角色用户权限RBAC

    关于角色.用户.权限的表设计,已经成为经典中的经典,到处都在用,这里只是收录一下: 角色表:{role_id,role_name} 用户表:{user_id,user_name} 权限表:{permi ...

  7. IDE神器intellij idea的基本使用 (转载)

    一.关于新建工程,导入工程,配置jdk,tomcat这里不做过多的讲述,必定网络上关于此类配置一堆一堆的. 二.编码快捷键(比较常用的快捷键)该套快捷键选择的是:Mac OS X 10.5+ 1. a ...

  8. Linux安装phpMywind

    1.安装MySQL http://www.cnblogs.com/wangshuyi/p/6091244.html 2.安装apache.php.及其扩展 yum install -y httpd p ...

  9. Java的访问权限(public并不等于默认)

    一共有四种访问权限,对应四个范围 1.private :只有本类内可以使用,即使是子类也没权使用 2.protect :子类和友好类能够使用,继承中经常用到 3.默认值,(就是什么都没写),只有同包名 ...

  10. Gradle中的SourceSet理解

    对于maven项目来说,目录结构是固定的,也就是像这样: src/main/ src/main/java/ src/main/resources/ src/test/ src/test/java/ s ...