javascript代码复用(四)-混入、借用方法和绑定
这篇继续说js的现代复用模式:混入、借用方法和绑定。
混入
可以针对前面提到的通过属性复制实现代码复用的想法进行一个扩展,就是混入(mix-in)。混入并不是复制一个完整的对象,而是从多个对象中复制出任意的成员并将这些成员组合成一个新的对象。
混入的实现并不难,只需要遍历每个参数,并且复制出传递给这个函数的每个对象中的每个属性。
function mix(){
var arg,prop,child={};
for(arg=0;arg<arguments.length;arg++){
for(prop in arguments[arg]){
if(arguments[arg].hasOwnProperty(prop)){
child[prop] = arguments[arg][prop];
}
}
}
return child;
}
现在,有了一个通用的mix-in函数,可以向它传递任意数量的对象,其结果将获得一个具有所有源对象属性的新对象,一个调用的例子:
var cake = mix(
{eggs : 2,large : true},
{butter : 1,sorted : true},
{flour : "3 cups"},
{suger : "sure!"}
);
console.dir(cake);
下面是控制台的输出:
butter 1
eggs 2
flour "3 cups"
large true
sorted true
sugar "sure!"
借用方法
有时可能恰好仅需要现有对象其中的一个或两个方法,在想要重用方法的同时,又不希望和源对象是父子的继承关系,也就是只想使用所需要的方法,而不需要那些永远用不到的其他方法。这种情况下,可以使用借用方法(borrowing method)来实现,即使用call()和apply(),区别就是传参的区别。
下面是一个例子,借用了数组的方法:
function f(){
var args = [].slice.call(arguments,1,3);
return args;
}
f(1,2,3,4,5,6);//[2,3]
其中创建空数组是为了使用数组的slice方法,也可以从Array的原型中借用方法,即Array.prototype.slice.call,这个需要输更长的字符,但是可以节省创建一个空数组的工作。
借用方法,不是通过call()和apply()就是通过简单的赋值,在借用方法的内部,this所指向的对象是基于调用表达式而确定的,但更多时候,最好可以锁定this的值,或者把它绑定到特定对象并预先确定该对象。
参考下面的例子,one对象有一个say()的方法:
var one = {
name : "object",
say : function(greet){
return greet+","+this.name;
}
};
one.say("hi");//"hi,object"
另一个对象two中没有say方法,但是可以从one那里借用:
var two = {
name : "another object"
};
one.say.apply(two,["hello"]);//"hello,another object"
上面借用的say()方法中的this指向了two,所以this.name是"another object".但是在什么场景中,应该给函数指针赋值一个全局变量,或者将函数作为回调函数传递?在程序中有这样的应用,并且出现了问题。
var say = one.say;
say("hoho");//"hoho,undefined" var yetanother = {
name : "Yet another object",
method : function(callback){
return callback("Hola");
}
};
yetanother.method(one.say);//"Hola,undefined"
在上面两种情况下this都指向了全局对象,并且代码都没有按预期运行。为了绑定对象与方法之间的关系,可以用下面的一个简单的函数:
function bind(o,m){
return function(){
return m.apply(o,arguments);
};
}
bind()接受了一个对象o和一个方法m,并将它们绑定起来,然后返回另一个函数。返回的函数可以通过闭包来访问o和m。所以在bind()返回后仍然可以访问o和m.可以使用bind()创建一个新函数:
var twosay = bind(two,one.say);
twosay("yo");//"yo,another object"
无论怎么调用twosay(),这个方法总是绑定到对象two上。
ES5中的bind()
ES5将bind()添加到Function.prototype,使得bind()像call()apply()一样易用。可以执行下面的表达式:
var newFunc = obj.someFunc.bind(myobj,1,2,3);
就是将someFunc()与myobj绑定到一起,并填充someFunc()的前3个参数。
在不支持ES5的环境下面运行的时候,看看怎么实现Function.prototype.bind():
if (typeof Function.prototype.bind === "undefined"){
Function.prototype.bind = function(thisArg){
var fn = this,
slice = Array.prototype.slice,
args = slice.call(arguments,1);
return function(){
return fn.apply(thisArg,args.concat(slice.call(arguments)));
};
};
}
它拼接了参数列表,即传给bind()的参数(第一个除外),以及那些传给由bind()返回新函数的参数,新函数将在后面调用。一个调用例子:
var twosay2 = one.say.bind(two);
twosay2("Bonjour");//"Bonjour,another object"
也可以传递一个参数:
var twosay3 = one.say.bind(two,"Nihao");
twosay3();//"Nihao,another object"
小结
在javascript中可能并不会像C#或Java一样经常面临继承的问题,一些原因是js库用一些方法解决了这个问题,另一些原因是在js中很少需要建立长而且复杂的继承链。在静态强类型语言中,继承可能是唯一复用代码的方法,但在js中经常有更简洁并且优雅的方法,包括借用方法,绑定,复制属性,及从多个对象中混入属性等方法。毕竟,代码重用才是最终目的,继承只是实现这个目标的方法之一。
--end--
javascript代码复用(四)-混入、借用方法和绑定的更多相关文章
- 编写高质量JavaScript代码的68个有效方法
简介: <Effective JavaScript:编写高质量JavaScript代码的68个有效方法>共分为7章,分别涵盖JavaScript的不同主题.第1章主要讲述最基本的主题,如版 ...
- javascript代码复用模式
代码复用有一个著名的原则,是GoF提出的:优先使用对象组合,而不是类继承.在javascript中,并没有类的概念,所以代码的复用,也并不局限于类式继承.javascript中创建对象的方法很多,有构 ...
- javascript代码复用模式(二)
前面说到,javascript的代码复用模式,可分为类式继承和非类式继承(现代继承).这篇就继续类式继承. 类式继承模式-借用构造函数 使用借用构造函数的方法,可以从子构造函数得到父构造函数传任意数量 ...
- javascript代码复用模式(三)
前面谈到了javascript的类式继承.这篇继续部分类式继承,及一些现代继承. 类式继承模式-代理构造函数 这种模式通过断开父对象与子对象之间原型之间的直接链接关系,来解决上次说到的共享一个原型所带 ...
- javascript代码复用--继承
由于javascript没有类的概念,因此无法通过接口继承,只能通过实现继承.实现继承是继承实际的方法,javascript中主要是依靠原型链要实现. 原型链继承 原型链继承是基本的继承模式,其本质是 ...
- javascript第十四课,方法的扩展prototype
所谓扩展方法就是,在原函数的基础上我们往对象里面添加一些自己需要的方法,例如: string对象 string.prototype.checkEmail=function(){ //方法体 //在这里 ...
- 【JavaScript代码实现四】获取和设置 cookie
// 创建cookie function setCookie(name, value, expires, path, domain, secure) { var cookieText = encode ...
- 《JavaScript 模式》读书笔记(6)— 代码复用模式3
我们之前聊了聊基本的继承的概念,也聊了很多在JavaScript中模拟类的方法.这篇文章,我们主要来学习一下现代继承的一些方法. 九.原型继承 下面我们开始讨论一种称之为原型继承(prototype ...
- javaScript DOM编程经常使用的方法与属性
DOM是Document Object Model文档对象模型的缩写.依据W3C DOM规范,DOM是一种与浏览器,平台,语言无关的接口,使得你能够訪问页面其它的标准组件. Node接口的特性和方法 ...
随机推荐
- .net关于httpModules的应用示例
这三个对象我们在开发Asp.net程序时经常会用到,似乎很熟悉,但有时 候又不太确定.本文通过一个简单的例子来直观的比较一下这三个对象的使用. HttpModule:Http模块,可以在页面处理前后. ...
- Oracle Erp常用网站
2014-01-01 Created By BaoXinjian
- php pcntl 多进程学习
1.捕获子进程退出(监听SIGCHLD信号,然后调用 pcntl_wait 函数) declare(ticks=1); pcntl_signal(SIGCHLD, "sig_handler& ...
- wcf Svcutil用法
[转] WCF中可以使用SVCUtil.exe生成客户端代理类和配置文件 1.找到如下地址“C:\Windows\System32\cmd.exe” 命令行工具,右键以管理员身份运行(视系统是否为w ...
- SteamVR Unity工具包(VRTK)之控制器交互
可交互对象(VRTK_InteractableObject) 可交互对象脚本被添加到需要用(如控制器)来交互的任何游戏对象上. 可用脚本参数如下 Touch Interactions 触摸交互 ...
- js手机短信按钮倒计时
/* 120秒手机短信按钮倒计时 */ exports.sendmessage = function (name) { var second = 120; $(name). ...
- Jquery Validation 插件验证手机号
自定义手机号验证代码 http://www.2cto.com/kf/201505/402781.html // 手机号码验证 jQuery.validator.addMethod("isM ...
- 使用gson-1.6.jar解析json
package com.example.mars_2900_json01; import java.io.StringReader; import java.lang.reflect.Type; im ...
- PDF出力相关资料
http://itext.2136553.n4.nabble.com/iText-SetFieldProperty-method-not-working-for-some-parameters-set ...
- JS 中的 Window 对象
窗口对象的属性和方法: 在js最外层写的function可以还可以理解为window对象的一个方法.定义的变量也可以称之为window对象的一个属性.例如:window.alert("--- ...