转载自:https://www.cnblogs.com/youxin/p/3410185.html
方法链一般适合对一个对象进行连续操作(集中在一句代码)。一定程度上可以减少代码量,缺点是它占用了函数的返回值。
一、方法体内返回对象实例自身(this)
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 
 | function ClassA(){this.prop1 = null;
 this.prop2 = null;
 this.prop3 = null;
 }
 ClassA.prototype = {
 method1 : function(p1){
 this.prop1 = p1;
 return this;
 },
 method2 : function(p2){
 this.prop2 = p2;
 return this;
 },
 method3 : function(p3){
 this.prop3 = p3;
 return this;
 }
 }
 
 | 
定义了function/类ClassA。有三个属性/字段prop1,prop2,prop3,三个方法methed1,method2,method3分别设置prop1,prop2,prop3。
链式调用如下:
| 12
 
 | var obj = new ClassA();  obj.method1(1).method2(2).method3(3);
 
 | 
可以看到对obj进行了连续三次操作,只要愿意ClassA的N多方法都这样定义,调用链会一直延续。
该方式缺点是链方法唯一地绑定于一种对象类型(ClaaaA),按这种方式实现链式操作,每定义一个类,其方法体中都要返回this。第二种方式可以解决这个问题。
二、对象传入后每次调用返回函数自身
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 
 | 
 
 
 function singleChain1(obj){
 function chain(){
 if (arguments.length == 0){
 return chain.obj;
 }
 var methodName = arguments[0], methodArgs = [].slice.call(arguments,1);
 chain.obj[methodName].apply(chain.obj,methodArgs);
 return chain;
 }
 chain.obj = obj;
 return chain;
 }
 
 
 
 
 function singleChain2(obj){
 return function(){
 var Self = arguments.callee; Self.obj = obj;
 if(arguments.length==0){
 return Self.obj;
 }
 var methodName = arguments[0], methodArgs = [].slice.call(arguments,1);
 Self.obj[methodName].apply(Self.obj,methodArgs);
 return Self;
 }
 }
 
 
 
 
 function singleChain3(obj){
 return function(){
 var Self = arguments.callee; Self.obj = obj;
 if(arguments.length==0){
 return Self.obj;
 }
 Self.obj[arguments[0]].apply(Self.obj,[].slice.call(arguments,1));
 return Self;
 }
 }
 
 | 
使用:
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 
 | 
 
 
 function chain(obj){
 return function(){
 var Self = arguments.callee; Self.obj = obj;
 if(arguments.length==0){
 return Self.obj;
 }
 Self.obj[arguments[0]].apply(Self.obj,[].slice.call(arguments,1));
 return Self;
 }
 }
 
 function ClassB(){
 this.prop1 = null;
 this.prop2 = null;
 this.prop3 = null;
 }
 ClassB.prototype = {
 method1 : function(p1){
 this.prop1 = p1;
 },
 method2 : function(p2){
 this.prop2 = p2;
 },
 method3 : function(p3){
 this.prop3 = p3;
 }
 }
 
 | 
注意ClassB的method1,method2,method3中不再返回this了。
链式调用如下:
| 12
 
 | var obj = new ClassB();  chain(obj)('method1',4)('method2',5)('method3',6);
 
 | 
第一种方式3次调用后返回了对象自身,这里使用一个空”()”取回对象
| 12
 
 | var result = chain(obj)('method1',4)('method2',5)('method3',6)();
 
 | 
这种方式写类时方法体中无须返回this,且可以对任何对象进行链式调用。
接下来介绍YUI中Node类实现的链式调用方法。
在YUI3中,Node类的基础是Dom,很多Node类的方法都是调用Dom类的同名方法,如上面提到的setAttribute、setStyle等,
在Dom类源码中也未设置返回本对象,在Node类提供了importMethods方法来导入Dom中相同的方法并支持链式调用。示例代码如下:
| 12
 3
 4
 5
 6
 7
 8
 9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 
 | function Dom(id){
 this.dom = document.getElementById(id);
 }
 Dom.setStyle = function(node,name,value){
 node.dom.style[name] = value;
 }
 Dom.setAttribute = function(node,name,v){
 node.dom.setAttribute(name,v);
 }
 
 function Node(id){
 this.dom = document.getElementById(id);
 }
 
 Node.addMethod = function(method,fn){
 Node.prototype[method] = function(){
 var me = this;
 
 
 arguments = Array.prototype.slice.call(arguments);
 arguments.unshift(me);
 fn.apply(me,arguments);
 return me;
 }
 }
 
 Node.importMethods = function(host,methods){
 for(var i in methods){
 var m = methods[i];
 var fn = host[m];
 Node.addMethod(m,fn);
 }
 }
 
 var methods = ['setStyle','setAttribute'];
 
 Node.importMethods(Dom,methods);
 
 var n = new Node('log').setStyle('border','2px solid red').setAttribute('t',22);
 
 | 
在实际使用中,可以对原有的对象方法(如Dom中的方法)扩展到另一个类中(如Node类),在Node类中进行链式调用。当然也可以使用同样的方式(importMethods)不扩展而是覆盖Dom中的方法。