第5章 原型

5.1 原型属性

        function f(a,b){
return a*b;
};
// length 属性
f.length; //
// constructor 构造属性
f.constructor; // function Function() { [native code] }
// prototype 属性 初始值是空对象
typeof f.prototype; //"object"

5.1.1 利用原型添加方法和属性

        function f1(name,color){
// 使用this添加属性
this.name=name;
this.color=color;
this.whatAreYou=function(){
return 'I am a'+this.color+''+this.name;
};
}
// 使用prototype属性添加属性和方法
f1.prototype.price=100;
f1.prototype.rating=3;
f1.prototype.getInfo=function (){
return 'Rating: '+this.rating+',price: '+this.price;
};

5.1.2 使用原型的属性和方法

        function f1(name,color){
// 使用this添加属性
this.name=name;
this.color=color;
this.whatAreYou=function(){
return 'I am a'+this.color+''+this.name;
};
}
// 使用prototype属性添加属性和方法
f1.prototype.price=100;
f1.prototype.rating=3;
f1.prototype.getInfo=function (){
return 'Rating: '+this.rating+',price: '+this.price;
};
var f2=new f1('webcam','black');
f2.color; // "black"
f2.getInfo(); //"Rating: 3,price: 100"
// 修改prototype属性,由同一构造器创建的所有对象的prototype属性也都会同时发生改变(会影响在修改之前已经创建的对象)
f1.prototype.get=function(what){
return this[what];
}
f2.get('price'); //

5.1.4 利用自身属性重写原型属性

对象自身属性的优先级高于原型属性

       function F1(name){
this.name=name;
};
F1.prototype.name='Alen';
var f2=new F1('XiaoMing');
f2.name; // "XiaoMing" 不是 Alen
// 使用hasOwnProperty()判断一个属性是自身属性还是原型属性
f2.hasOwnProperty('name'); //true
//删除自身属性后,原型属性才会显示
delete f2.name; // true
f2.name; // "Alen"

5.1.5 isPrototypeOf()方法

isPrototypeOf():当前对象是否为另一个对象的原型

        var person={
like:'eat',
age:12,
sex:'boy'
};
function F1(name){
this.name=name;
}
F1.prototype=person;
var f2=new F1('Alen');
// 判断person是否为f2的原型
person.isPrototypeOf(f2); //true
//使用getPrototypeOf()获取原型
Object.getPrototypeOf(f2)===person; //true

5.1.6 神秘的__proto__链接

5.2 扩展内建对象

内建对象的构造器函数可以通过其原型来进行扩展。

     <script type="text/javascript">
// 示例1:扩展Array的内建函数,用来查询数组中是否存在某个特定的值
Array.prototype.inArray=function(needle){
for (var i = 0; i < this.length; i++) {
if(this[i]===needle){
return true;
}else{
return false;
}
};
};
var colors=['red','black','blue'];
colors.inArray('red'); //true
colors.inArray('green'); //false
//示例2:反转字符串
// 先利用split()将目标字符串转换成数组,然后调用该数组的reverse()方法产生一个反向数组。
// 最后通过join()方法将结果数组转换为字符串
String.prototype.reverse=function (){
return Array.prototype.reverse.apply(this.split('')).join('');
};
"bums".reverse(); //smub
</script>

5.2.1 关于内建对象的讨论

如果想通过原型为某个对象添加一个新属性,务必检查一下该属性是否已经存在。

5.2.2 原型陷阱

注意: 1> 当对原型对象执行完全替换时,可能会触发原型链中某种异常

2> prototype.constructor 属性不可靠

 <script type="text/javascript">
function Dog(){
this.tail=true;
};
var f1=new Dog();
var f2=new Dog();
// 仍可以为Dog()的原型添加属性,添加之前已经存在的对象也可访问新属性
Dog.prototype.say=function(){
return 'worf';
};
// f1、f2可以访问新方法
f1.say(); //"worf"
f2.say(); //"worf"
// 检查对象的构造器
f1.constructor === Dog; // true 正常
f2.constructor === Dog;// true 正常
Dog.prototype.constructor === Dog; // true // 用自定义新对象完全覆盖掉原有的原型对象
Dog.prototype={
paws:4,
hair:true
};
// 这样原有对象不能访问新属性,但通过__proto__与原有的原型保持联系
f1.paws; //undefined
f1.say(); //"worf"
typeof f1.__proto__.say; // "function"
// 之后创建的对象室友被更新后的prototype对象
var f3=new Dog();
f3.say(); // f3.say is not a function
f3.tail; // true
f3.paws; //
// __proto__指向新的原型对象
f3.__proto__.hair; //true // 新对象的constructor属性不在保持正确
f3.constructor;// function Object() { [native code] }
// 本应该指向Dog(),现在指向Object
f1.constructor; //function Dog(){this.tail=true;} 仍为Dog() 不变
Dog.prototype.constructor; // function Object() { [native code] }
//重新设置constructor属性 解决
Dog.prototype.constructor=Dog;
new Dog().constructor === Dog; // true
</script>

JavaScript面向对象编程指南(五) 原型的更多相关文章

  1. 「JavaScript面向对象编程指南」原型

    在 JS 中,函数本身也是一个包含了方法(如apply和call)和属性(如length和constructor)的对象,而prototype也是函数对象的一个属性 function f(){} f. ...

  2. 《JavaScript面向对象编程指南(第2版)》读书笔记(一)

    目录 一.对象 1.1 获取属性值的方式 1.2 获取动态生成的属性的值 二.数组 2.1 检测是否为数组 2.2 增加数组长度导致未赋值的位置为undefined 2.3 用闭包实现简易迭代器 三. ...

  3. 《JavaScript面向对象编程指南(第2版)》读书笔记(二)

    <JavaScript面向对象编程指南(第2版)>读书笔记(一) <JavaScript面向对象编程指南(第2版)>读书笔记(二) 目录 一.基本类型 1.1 字符串 1.2 ...

  4. 《JavaScript面向对象编程指南》读书笔记②

    概述 <JavaScript面向对象编程指南>读书笔记① 这里只记录一下我看JavaScript面向对象编程指南记录下的一些东西.那些简单的知识我没有记录,我只记录几个容易遗漏的或者精彩的 ...

  5. 《JavaScript面向对象编程指南》读书笔记①

    概述 JavaScript快忘完了,想看一本专业书拾遗,所以看了这本<JavaScript面向对象编程指南>. 个人觉得这本书讲的很透彻很易懂,一些原来有疑惑的地方在这本书里面豁然开朗,看 ...

  6. 闭包初体验 -《JavaScript面向对象编程指南》

    下面是我对闭包的理解:(把他们整理出来,整理的过程也是在梳理) 参考<JavaScript面向对象编程指南> 1.首先,在理解闭包之前: 我们首先应该清楚下作用域和作用域链 作用域:每个函 ...

  7. 《JavaScript面向对象编程指南》

    第一章.引言 1.5 面向对象的程序设计常用概念 对象(名词):是指"事物"在程序设计语言中的表现形式. 这里的事物可以是任何东西,我们可以看到它们具有某些明确特征,能执行某些动作 ...

  8. 《JavaScript面向对象编程指南》译者序

    相对于Perl.Python等动态脚本语言来说,JavaScript确实是一门饱受误解的语言.对于译者这种从20世纪90年代末走过来的C++程序员来说,尤其如此.在那个年代,提起JavaScript总 ...

  9. JavaScript面向对象编程指南

    引言 面向对象程序设计 基本数据类型.数组.循环及条件表达式 基本数据类型 函数 函数Function 预定义函数 变量的作用域 函数也是数据 闭包 对象 原型 原型 继承 原型链 浅拷贝与深拷贝 原 ...

随机推荐

  1. 软件测试人员需要掌握的linux命令(二)

    2 设备管理 2.1 mount 名称 : mount 使用权限 : 系统管理者或/etc/fstab中允许的使用者 使用方式 : mount [-hV] mount [-fnrsvw] [-t vf ...

  2. Oracle SQL调优记录

    目录 一.前言 二.注意点 三.Oracle执行计划 四.调优记录 @ 一.前言 本博客只记录工作中的一次oracle sql调优记录,因为数据量过多导致的查询缓慢,一方面是因为业务太过繁杂,关联了太 ...

  3. Python中函数和模块的体验与使用

    函数基础 目标 函数的快速体验 函数的基本使用 函数的参数 函数的返回值 函数的嵌套调用 在模块中定义函数 01. 函数的快速体验 1.1 快速体验 所谓函数,就是把 具有独立功能的代码块 组织为一个 ...

  4. Jexus使用的相关记录

    前言 本文是零零散散的记录,部分内容是我在平时工作中用到的,部分是从群里"偷"来的,所以难免会有一些错误. 主要还是希望能帮到部分使用Jexus的朋友. 安装 curl https ...

  5. SpringBoot配置Cors解决跨域请求问题

    一.同源策略简介 同源策略[same origin policy]是浏览器的一个安全功能,不同源的客户端脚本在没有明确授权的情况下,不能读写对方资源. 同源策略是浏览器安全的基石. 什么是源 源[or ...

  6. leetcode — jump-game

    /** * Source : https://oj.leetcode.com/problems/jump-game/ * * Created by lverpeng on 2017/7/17. * * ...

  7. Http系列目录

    1.Http简史 2.Http协议基本术语 3.Http1.1 4.Http2.0

  8. Android Navigation 架构组件入门教程

    Android Navigation 架构组件入门教程 版权声明:本文为博主原创文章,未经博主允许不得转载. 转载请表明出处:https://www.cnblogs.com/cavalier-/p/1 ...

  9. Pytorch1.0入门实战一:LeNet神经网络实现 MNIST手写数字识别

    记得第一次接触手写数字识别数据集还在学习TensorFlow,各种sess.run(),头都绕晕了.自从接触pytorch以来,一直想写点什么.曾经在2017年5月,Andrej Karpathy发表 ...

  10. Perl文件句柄相关常量变量

    文件句柄相关变量 对应的官方手册:http://perldoc.perl.org/perlvar.html#Variables-related-to-filehandles 默认情况下: $/:输入行 ...