JavaScript之美读书笔记一
前两天去图书馆借了一本《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之美读书笔记一的更多相关文章
- JavaScript 函数式编程读书笔记2
概述 这是我读<javascript函数式编程>的读书笔记,供以后开发时参考,相信对其他人也有用. 说明:虽然本书是基于underscore.js库写的,但是其中的理念和思考方式都讲的很好 ...
- JavaScript 函数式编程读书笔记1
概述 这是我读<javascript函数式编程>的读书笔记,供以后开发时参考,相信对其他人也有用. 说明:虽然本书是基于underscore.js库写的,但是其中的理念和思考方式都讲的很好 ...
- javascript高级程序设计读书笔记-事件(一)
读书笔记,写的很乱 事件处理程序 事件处理程序分为三种: 1.html事件2. DOM0级,3,DOM2级别 没有DOM1 同样的事件 DOM0会顶掉html事件 因为他们都是属性 而 ...
- javascript框架设计(读书笔记)
我觉得多看几本进阶的书 与其十本书读一遍,不如一本书读十遍 读书的启示: 读好书(看推荐) 精读(重复看) 能读厚书(javascript权威指南) Object.keys Object.keys=O ...
- <JavaScript语言精粹>--<读书笔记三>之replace()与正则
今天有人问我repalce(),他那个题目很有意思.我也不会做,于是我就去查,结果发现就是最基础的知识的延伸. 所以啊最基础的知识才是很重要的,千万不能忽略,抓起JS就写代码完全不知到所以然,只知道写 ...
- <JavaScript语言精粹>-读书笔记(一)
用object.hasOwnProperty(variable)来确定这个属性名是否为该对象成员,还是来自于原型链. for(my in obj){ if(obj.hasOwnProperty(my) ...
- JavaScript语言精粹读书笔记 - JavaScript函数
JavaScript是披着C族语言外衣的LISP,除了词法上与C族语言相似以外,其他几乎没有相似之处. JavaScript 函数: 函数包含一组语句,他们是JavaScript的基础模块单元,用于代 ...
- 深入理解javascript系列,读书笔记
深入理解JavaScript系列(2):揭秘命名函数表达式 1.讲了函数声明和函数表达式的区别,包括一些在函数提升上的区别 2.如果给函数表达式的函数也取名,会在调试的时候受益 3.不要在block( ...
- JavaScript高级程序设计 读书笔记
第一章 JavaScript 简介 第二章 Html中使用JavaScript 第三章 基本概念 第四章 变量,作用域,内存 第五章 引用类型 第六章 面向对象 第七章 函数表达式 第八章 BOM 第 ...
随机推荐
- hive表链接
等值连接 不等职链接 外部链接 没有包含在聚合函数(这里是count)中的列,都需要包含在group by函数中: 正确的外链接的写法,用的是右外链接: 自链接表 把同一张表 看成了2张表
- Alertmanager 安装(k8s报警)
一.下载Alertmanager https://prometheus.io/download/ wget https://github.com/prometheus/alertmanager/rel ...
- centos无网络问题
- How to enable usb on vbox
Device-->Install Guest Addition Shared Folders Settings-->Advanced-->Shared Clipboard--> ...
- Codeforces Round #545 Div. 1自闭记
A:求出该行该列各有多少个比其小的取max,该行该列各有多少个比其大的取max,加起来即可. #include<iostream> #include<cstdio> #incl ...
- vue——script内容详解
<script> export default { name: "Home", data() { return {}; }, methods: { // 组件的方法 } ...
- HDU5384-Hotaru's problem-Manacher
找出紧挨的三个回文串,例如abccbaabc ,形如ABA格式,其中AB为回文串.计算最长的长度. 首先用Manacher处理回文半径.然后就是找到两个点,都是偶数的回文串,并且共享了中间一段. 之后 ...
- POJ 3264-Balanced Lineup-RMQ问题
裸RMQ问题 #include <cstdio> #include <algorithm> #include <cstring> using namespace s ...
- 数据库 -- pymysql
pythen3连接mysql pymsql介绍 PyMySQL 是在 Python3.x 版本中用于连接 MySQL 服务器的一个库,Python2中则使用mysqldb. Django中也可以使用P ...
- Codeforces963C Cutting Rectangle 【数学】
错了一个小地方调了一晚上.... 题目大意: 给出最多2E+5种不同的矩形,每种有它的长h和宽v还有数量d,现在你要构造大矩形,使得在上面沿着平行于长或宽的边划刀,切出来的矩形正好是给出的所有矩形.问 ...