前两天去图书馆借了一本《JavaScript之美》,在书架上无意中看到的,想着那就看看吧。

第一章

原型

有些JavaScript爱好者宣称JavaScript是一种基于原型而不是面向对象的语言,任何带有“类”字样的方法根本不适用于JavaScript。但“原型”的含义是什么?原型和类有着怎样的区别?

用通用的编程术语来讲,原型是指为其他对象提供基本行为的对象。其他对象也可在此基础上扩展基本行为,加入个性化行为。该过程也称为有差异的继承,有别于类继承的是它不需要明确指定类型(静态或动态),从形式上而言,它也不是在一种类型的基础上定义其他类型。类继承的目的是复用,而原型继承则不一定。

JavaScript的每个对象均指向一个原型对象并继承其属性。JavaScript的原型是实现复用的好工具。原型还可以继承自其他原型,从而形成原型链。

JavaScript将prototype属性绑定到构造器,其结果是多个层级的对象继承通常需要链接构造器和原型来实现。构造器-原型链句法,不但不优雅,缺点还体现在需要预先规划。ES6的class关键字只是将现有实现方式形式化。

ES5标准引入了Object.create,以增加原型继承的灵活度,扩展应用场景。该方法允许将原型直接赋给对象,JavaScript原型不再受限于构造器和类型的限制。

var circle = Object.create({
area: function() {
return Math.PI * this.radius * this.radius;
},
grow: function() {
this.radius++;
},
shrink: function() {
this.radius--;
}
});

Object.create方法的第2个参数可选,表示继承自哪个对象。不幸的是第2个参数不是对象自身,而是一个完整的 meta 属性定义:

var circle = Object.create({
area: function() {
return Math.PI * this.radius * this.radius;
},
grow: function() {
this.radius++;
},
shrink: function() {
this.radius--;
}
}, {
radius: {
writable: true, configurable: true, value: 7
}
});

或者可以手动将属性赋给它。

即便如此,Object.create方法也只是允许对象继承某个原型的属性,但真实应用场景,往往需要从多个原型对象获得行为。

mixin方法

函数复用的最基本方法是手动委托,任何公共函数都可以直接用call或apply方法调用。

mixin基础

从传统意义上讲,mixin是一个类,它定义了一组原本要用实体定义的函数。然而,mixin类被视作是抽象的,因为它不是由自己来完成实例化。相反,具体的类通过复制(或借)mixin类的函数,继承mixin的行为,而不必跟行为的提供者产生正式的关系。mixin类可以是常规对象,原型或函数等。

应用场景

1:类形式的mixin

先定义一个圆形mixin

var circleFns = {
area: function() {
return Math.PI * this.radius * this.radius;
},
grow: function() {
this.radius++;
},
shrink: function() {
this.radius--;
}
};

再看一个定义按钮行为的mixin

var clickableFns = {
hover: function() {
console.log('hovering');
},
press: function() {
console.log('button pressed');
},
fire: function() {
return this.action();
}
};

如何将mixin对象整合到你的对象之中?你需要借助extend函数。

function extend(destination, source) {
for(var key in source) {
if(source.hasOwnProperty(key)) {
destination[key] = source[key];
}
}
return destination;
}

用刚才创建的两个mixin对象,扩展新对象RoundButton的基础原型RoundButton.prototype:

var RoundButton = function(radius, label, action) {
this.radius = radius;
this.label = label;
this.action = action;
}; extend(RoundButton.prototype, circleFns);
extend(RoundButton.prototype, clickableFns); var roundButton = new RoundButton(3, 'send', function(){return 'send';}); roundButton.area();
roundButton.fire();

函数形式的mixin

下面我们将圆形和按钮mixin改写为函数。

var circleFns = function() {
this.area = function() {
return Math.PI * this.radius * this.radius;
};
this.grow = function() {
this.radius++;
};
this.shrink = function() {
this.radius--;
};
}; var clickableFns = function() {
this.hover = function() {
console.log('hovering');
};
this.press = function() {
console.log('button pressed');
};
this.fire = function() {
return this.action();
};
};

现在原型对象通过Function.prototype.call就能将自己注入到目标对象中去

circleFns.call(RoundButton.prototype);
clickableFns.call(RoundButton.prototype);

这种方法给人的感觉是很贴切,编码风格自然简洁,this总是指向接收者而不是我们不需要的抽象对象,并且,我们不必提防无意中复制了被继承的属性。

带options参数

函数形式的mixin方法还支持通过options参数将行为参数化,以掺杂使用各种行为。看下述示例:

var withOval = function(options) {
this.area = function() {
return Math.PI * this.longRadius * this.shortRadius;
};
this.ratio = function() {
return this.longRadius / this.shortRadius;
};
this.grow = function() {
this.shortRadius += (options.growBy / this.ratio());
this.longRadius += (options.growBy);
};
this.shrink = function() {
this.shortRadius -= (options.shinkBy / this.ratio());
this.longRadius -= (options.shinkBy);
};
}; var OvalButton = function(longRadius, shortRadius, label, action) {
this.longRadius = longRadius;
this.shortRadius = shortRadius;
this.label = label;
this.action = action;
}; withOval.call(OvalButton.prototype, {growBy: 2, shinkBy: 2}); button.area();
button.grow();
button.area();

添加缓存

函数形式的mixin还能做进一步优化,对mixin构造闭包,我们能缓存第一次定义时的结果,由此所带来的性能上的提升非常显著。下面是增加缓存机制的withRectangle mixin:

var withRectangle = (function() {
function area() {
return this.length * this.width;
};
function grow() {
this.length++, this.width++;
};
function shrink() {
this.length--, this.width--;
};
return function() {
this.area = area;
this.grow = grow;
this.shrink = shrink;
return this;
};
})(); var RectangleButton = function(length, width) {
this.length = length;
this.width = width;
};
withRectangle.call(RectangleButton.prototype); var button = new RectangleButton(4, 2);
button.area();

小结

类继承重复用一个对象定义另一个对象,由此形成了一系列紧密的耦合关系,将不同的层级粘结在一起,对象之间的依赖关系非常复杂。相反,mixin极其敏捷,你几乎不需要调整代码库,只要发现了一组通用的,可共享的行为,就可以根据需要创建mixin,而其他所有对象不管在整个模型中扮演什么角色,都可以访问mixin的功能。mixin和对象之间的关系非常自由:mixin的任意组合可应用于任意对象,对象对应用于它的mixin数量也没有限制。这正是原型继承赋予我们的根据机会复用代码的能力。

JavaScript之美读书笔记一的更多相关文章

  1. JavaScript 函数式编程读书笔记2

    概述 这是我读<javascript函数式编程>的读书笔记,供以后开发时参考,相信对其他人也有用. 说明:虽然本书是基于underscore.js库写的,但是其中的理念和思考方式都讲的很好 ...

  2. JavaScript 函数式编程读书笔记1

    概述 这是我读<javascript函数式编程>的读书笔记,供以后开发时参考,相信对其他人也有用. 说明:虽然本书是基于underscore.js库写的,但是其中的理念和思考方式都讲的很好 ...

  3. javascript高级程序设计读书笔记-事件(一)

    读书笔记,写的很乱   事件处理程序   事件处理程序分为三种: 1.html事件2. DOM0级,3,DOM2级别  没有DOM1 同样的事件 DOM0会顶掉html事件   因为他们都是属性  而 ...

  4. javascript框架设计(读书笔记)

    我觉得多看几本进阶的书 与其十本书读一遍,不如一本书读十遍 读书的启示: 读好书(看推荐) 精读(重复看) 能读厚书(javascript权威指南) Object.keys Object.keys=O ...

  5. <JavaScript语言精粹>--<读书笔记三>之replace()与正则

    今天有人问我repalce(),他那个题目很有意思.我也不会做,于是我就去查,结果发现就是最基础的知识的延伸. 所以啊最基础的知识才是很重要的,千万不能忽略,抓起JS就写代码完全不知到所以然,只知道写 ...

  6. <JavaScript语言精粹>-读书笔记(一)

    用object.hasOwnProperty(variable)来确定这个属性名是否为该对象成员,还是来自于原型链. for(my in obj){ if(obj.hasOwnProperty(my) ...

  7. JavaScript语言精粹读书笔记 - JavaScript函数

    JavaScript是披着C族语言外衣的LISP,除了词法上与C族语言相似以外,其他几乎没有相似之处. JavaScript 函数: 函数包含一组语句,他们是JavaScript的基础模块单元,用于代 ...

  8. 深入理解javascript系列,读书笔记

    深入理解JavaScript系列(2):揭秘命名函数表达式 1.讲了函数声明和函数表达式的区别,包括一些在函数提升上的区别 2.如果给函数表达式的函数也取名,会在调试的时候受益 3.不要在block( ...

  9. JavaScript高级程序设计 读书笔记

    第一章 JavaScript 简介 第二章 Html中使用JavaScript 第三章 基本概念 第四章 变量,作用域,内存 第五章 引用类型 第六章 面向对象 第七章 函数表达式 第八章 BOM 第 ...

随机推荐

  1. 浅谈基于Prism的软件系统的架构设计

    很早就想写这么一篇文章来对近几年使用Prism框架来设计软件来做一次深入的分析了,但直到最近才开始整理,说到软件系统的设计这里面有太多的学问,只有经过大量的探索才能够设计出好的软件产品,就本人的理解, ...

  2. Java多线程5:Synchronized锁机制

    一.前言 在多线程中,有时会出现多个线程对同一个对象的变量进行并发访问的情形,如果不做正确的同步处理,那么产生的后果就是“脏读”,也就是获取到的数据其实是被修改过的. 二.引入Synchronized ...

  3. django学习自修第一天【简介】

    1. MVC框架 MVC框架的核心思想是解耦,降低各功能之间的耦合性,方便重构代码 (1)低耦合,高内聚 (2)高可扩展性 (3)向后兼容 2. MVT框架 V(视图):核心处理,接受请求,调用模型获 ...

  4. vue監聽屬性

    使用$watch,就是監聽到某個值發生變化,執行回調函數.

  5. JAVA 变量 数据类型 运算符 知识小结

    ---------------------------------------------------> JAVA 变量 数据类型 运算符 知识小结 <------------------ ...

  6. elasticsearch索引合并

    参考地址:http://cwiki.apachecn.org/display/Elasticsearch/Reindex+API 1.首先插入准备数据,创建两个索引. (1).PUT  http:// ...

  7. 前端base64、baseurl加解密和RSA加解密

    由于项目最近要进行安全测试,前端的用户和密码都是明文数据传送给后台那里,其实这样很很不安全的,容易泄露个人信息和密码.中间服务器的同事就提出,可以通过前端接收公钥,利用公钥对密码进行加密,把加密过密码 ...

  8. Educational Codeforces Round 60 Div. 2

    F:考虑对于每个字母对求出删掉哪些字符集会造成字符串不合法,只要考虑相邻出现的该字母对即可,显然这可以在O(np2)(或小常数O(np3))内求出.然后再对每个字符集判断是否能通过一步删除转移而来即可 ...

  9. python基础成长之路三

    1,基础数据类型 总览 int :数字  用于计数,计算,运算等...1 , 2 , 3  , 100 , ... str :字符串  用户少量的数据储存,便于操作   "这就是字符串&qu ...

  10. Python3.x标准模块库目录

    文本 string:通用字符串操作 re:正则表达式操作 difflib:差异计算工具 textwrap:文本填充 unicodedata:Unicode字符数据库 stringprep:互联网字符串 ...