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. LCS and LIS

    LCS #include<bits/stdc++.h> using namespace std; typedef long long ll; int n,m; char s[1005],t ...

  2. (转)mysql 备份与恢复mysqlhotcopy

    原文:http://fuwenchao.blog.51cto.com/6008712/1331910 mysqlhotcopy是一个Perl脚本,最初由Tim Bunce编写并提供.它使用LOCK T ...

  3. spring使用@Value注解读取.properties文件时出现中文乱码问题的解决

    解决办法 在spring中我们常常使用.properties对一些属性进行一个提前配置, spring 在读取*.properties文件时, 默认使用的是asci码, 这时 我们需要对其编码进行转换 ...

  4. JavaScript数据结构-7.链表

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  5. SpringCloud - RestTemplate 的三种使用方式

    1. 直接使用 RestTemplate restTemplate = new RestTemplate(); String result = restTemplate.getForObject(&q ...

  6. git基本操作若干

    开发合并分支,推送到远程1. git diff 查看修改内容: . git add -A 添加所有修改: . git commit -am "给人的提示"; 提交修改内容 . gi ...

  7. json和xml以及ajax的数据格式用法

    JSON的两个方法: 1.将字符串转换为JSON格式:parse(). 2.将原生JavaScript值转换为JSON字符串:stringify(); <!DOCTYPE html> &l ...

  8. [转]Repeat Page Header on each Page for reports SSRS

    本文转自:https://stackoverflow.com/questions/3475144/i-want-to-repeat-page-header-on-each-page-for-repor ...

  9. [转]使用ASP.NET Web API 2创建OData v4 终结点

    本文转自:http://www.cnblogs.com/farb/p/ODataAspNetWebAPI.html 开放数据协议(Open Data Protocol[简称OData])是用于Web的 ...

  10. Gauva的安装——入门篇

    Guava工程包含了若干被Google的 Java项目广泛依赖 的核心库,例如:集合 [collections] .缓存 [caching] .原生类型支持 [primitives support] ...