const定义的常量,一般是不能修改的。

比如:

 const TIME_OUT = 10000;

但是当值为引用类型值时,还是可以操作对象,扩展或修改对象属性、方法等等。

以下演示代码的操作是不会报错,且行之有效的。。

 const person = {
name: 'xm'
}
person.age = 18;
person.name = 'xh';

ES6(?)在Object上添加了一个静态方法freeze() ,可以禁止修改、扩展引用类型值。

用法就是把person作为参数传入该方法中。如:

 const person = {
name: 'xm'
}
console.log(person);// {name: 'xm'};
Object.freeze(person);
person.age = 18;
person.name = 'xh';
console.log(person);// {name: 'xm'};

好厉害,这下子就成了名副其实的常量了。emm....

freeze方法实现原理的简单模拟

要使用的到方法包括Object.definedProperty()、Object.seal()

Object.definedProperty()方法可以定义对象的属性的特性。如可不可以删除、可不可以修改、访问这个属性的时候添油加醋等等。。

用法(详见高级程序设计P139):

 Object.defineProperty(person, 'name', {
configurable: false,// 表示能否通过delete删除属性,能否修改属性的特性...
enumerable: false,// 表示是否可以枚举。直接在对象上定义的属性,基本默认true
writable: false,// 表示能否修改属性的值。直接在对象上定义的属性,基本默认true
value: 'xm'// 表示属性的值。访问属性时从这里读取,修改属性时,也保存在这里。
})

通过以上代码的设置,name属性就变成了不能删除、不可重新修改特性、不可枚举、不能修改的属性值的属性了。

测试结果:

非常重要的一个方法。以上就是它的简单用法以及实现效果。

Object.seal()方法可以让对象不能被扩展、删除属性等等。用法:Object.seal(person);

有了这两个方法,就可以实现一个简单的freeze()方法了。代码演示:

 function myFreeze(obj) {
if (obj instanceof Object) {
Object.seal(obj);
let p;
for (p in obj) {
if (obj.hasOwnProperty(p)) {
Object.defineProperty(obj, p, {
writable: false
});
myFreeze(obj[p]);// 递归,实现更深层次的冻结
}
}
}
}

以上函数先判断传入参数是否为Object类型。为真,接着封闭对象。for in 循环遍历对象。

剔除原型属性,设置属性的writable特性为false。递归调用该函数,传入属性值。

测试结果:

学习完一个。再接再厉。

来实现一下instanceof 操作符吧

instanceof的大致效果是:当左边是基本类型值时,一律返回false。 当左边是引用类型值时,如果右边的原型对象,在左边的原型链上存在,返回真,否则假。

有了大概的使用效果,加上一些方法。就可以实现一个简单的instanceof操作符了。代码演示:

 function myInstanceof(leftVal, rightFunc) {
if (typeof rightFunc !== 'function') throw new Error('第二个参数请传入构造函数名');
return rightFunc.prototype.isPrototypeOf(leftVal);
// F.prototype.isPrototypeOf(obj)判断obj的原型指针是否指向传入构造函数的原型对象,这个过程会往上层层判断。
// 比如,以下验证myInstanceof(xm, Object);xm的原型指针指向Person.prototype,Person.prototype的原型指针指向Object.prototype。所以返回true。 4 }

验证结果:

其他一些实现方法:

 function myInstanceof(leftVal, rightFunc) {
if (typeof rightFunc !== 'function') throw new Error('第二个参数请传入构造函数名');
if (typeof leftVal !== 'object' || leftVal === null) return false;
if (leftVal.__proto__ === rightFunc.prototype) {// __proto__原型指针
return true;
} else {
return myInstanceof(leftVal.__proto__, rightFunc);
}
}

以上利用浏览器在对象上布置的__proto__属性、加递归调用实现判断。

 function myInstanceof(leftVal, rightFunc) {
if (typeof rightFunc !== 'function') throw new Error('第二个参数请传入构造函数名');
if (typeof leftVal !== 'object' || leftVal === null) return false;
let _proto = Object.getPrototypeOf(leftVal);
while (_proto) {
if (_proto === rightFunc.prototype) {
return true;
}
_proto = Object.getPrototypeOf(_proto);
}
return false;
}

以上利用Object.getPrototypeOf()方法获取对象的原型,利用while循环层层递进。都没有,最后Object.getPrototypeOf()值为null。

最新文章

  1. 通过3个Hello World应用来了解ASP.NET 5应用是如何运行的(1)
  2. A ship is always safe at the shore - but that is not what it is built for.
  3. OC第九节——协议与代理
  4. android开发 wifi开发工具类
  5. Weblogic12c安装与配置详解
  6. boost------function的使用(Boost程序库完全开发指南)读书笔记
  7. [Cycle.js] Fine-grained control over the DOM Source
  8. Delphi获取当前系统时间(使用API函数GetSystemTime)
  9. Tosska SQL Tuning Expert 工具优化SQL语句
  10. js-模块化(三大模块化规范)
  11. fortran常用语句--读写带注释文档、动态数组等语法
  12. python笔记13-文件读写
  13. CNN tricks
  14. python装饰器 练习
  15. c# 生成随机时间
  16. 【数据库】mysql的安装
  17. git修改用户名和邮箱
  18. JS 对象 合并
  19. Emscripten编译环境搭建--将C和C++编译成JS
  20. memset用法详解

热门文章

  1. mtcnn论文学习
  2. 【Python】使用POST方式抓取有道翻译结果
  3. Qt Http get
  4. python编程中的一些有用插件或工具
  5. 数据标记系列——图像分割 & Curve-GCN
  6. php mkdir没有权限不能创建成功的问题
  7. 【linux学习笔记一】目录处理命令
  8. 图解JavaScript闭包面试题
  9. C# Hook原理及EasyHook
  10. ROS初探--意义、基本模块