对于组合设计模式:

 (1)组合模式中把对象分为两种(组合对象,和叶子对象)
(2)组合对象和叶子对象实现:同一批操作
(3)对组合对象执行的操作可以向下传递到叶子节点进行操作
(4)这样就会弱化类与类之间的耦合
(5)他常用的手法是把对象组合成属性结构的对象
根据组合模式的这些特性我们改写代码如下:
由于用到了接口检验所以我们先引入接口文件代码
//定义一个静态方法来实现接口与实现类的直接检验
//静态方法不要写出Interface.prototype ,因为这是写到接口的原型链上的
//我们要把静态的函数直接写到类层次上
//定义一个接口类
var Interface=function (name,methods) {//name:接口名字
if(arguments.length<){
alert("必须是两个参数")
}
this.name=name;
this.methods=[];//定义一个空数组装载函数名
for(var i=;i<methods.length;i++){
if(typeof methods[i]!="string"){
alert("函数名必须是字符串类型");
}else {
this.methods.push( methods[i]);
}
}
};
Interface.ensureImplement=function (object) {
if(arguments.length<){
throw new Error("参数必须不少于2个")
return false;
}
for(var i=;i<arguments.length;i++){
var inter=arguments[i];
//如果是接口就必须是Interface类型
if(inter.constructor!=Interface){
throw new Error("如果是接口类的话,就必须是Interface类型");
}
//判断接口中的方法是否全部实现
//遍历函数集合
for(var j=;j<inter.methods.length;j++){
var method=inter.methods[j];//接口中所有函数 //object[method]传入的函数
//最终是判断传入的函数是否与接口中所用函数匹配
if(!object[method]||typeof object[method]!="function" ){//实现类中必须有方法名字与接口中所用方法名相同
throw new Error("实现类中没有完全实现接口中的所有方法")
}
}
}
}

(1)统一接口

  var composite=new Interface("composite",["getChildByName","add"]);//侧重点获取子
var student=new Interface("composite",["goToClass","finishClass"]);//侧重点为每个对象的实现

(2)定义组合类

 var compositeObj=function (name) {
this.name=name;
this.type="com";//默认是组合类
var childs=new Array();
//得到相关的所有孩子节点
this.getChildByName=function (name) {
//涉及到递归
var toChilds=new Array();
if(!name){
for(var i=;i<childs.length;i++){
if(childs[i].type=="com"){
toChilds=toChilds.concat(childs[i].getChildByName())
}else {
toChilds.push(childs[i]);
}
}
}else {
for (var i = ; i < childs.length; i++){
if(childs[i].name==name){
if(childs[i].type=="com"){
toChilds=toChilds.concat(childs[i].getChildByName());
break;
}else {
toChilds.push(childs[i]);
break;
}
}else {
if(childs[i].type == "com"){
toChilds =toChilds.concat(childs[i].getChildByName(name));
}
}
}
}
return toChilds;
};
//增加子节点
this.add=function (child) {
childs.push(child);
return this;
};
//去上课
this.goToClass=function (name) {
var toChilds=this.getChildByName(name);
for(var i=;i<toChilds.length;i++){
toChilds[i].goToClass();
}
};
//下课
this.finishClass=function (name) {
var toChilds=this.getChildByName(name);
for(var i=;i<toChilds.length;i++){
toChilds[i].finishClass();
}
};
Interface.ensureImplement(this,composite,student);
};

(3)定义叶子类

    var studentObj=function (name) {
this.name=name;
this.type="student";//默认是叶子
//得到所有孩子节点
this.getChildByName=function (name) {
if(this.name==name){
return this;
}else {
return null;
}
}
//增加子节点
this.add=function (child) {
throw new Error("add 不成被初始化(在叶子了中)")
}
//去上课
this.goToClass = function(name){
document.write(this.name +" 去上课<br>");
}
//下课
this.finishClass = function(name){
document.write(this.name +" 下课<br>");
}
Interface.ensureImplement(this,composite,student);
}

(4)应用---将学校,班级,组,学生关联起来

 var  astudent=new studentObj("我是a同学");
var bstudent=new studentObj("我是b同学");
var cstudent=new studentObj("我是c同学");
var dstudent=new studentObj("我是d同学");
var estudent=new studentObj("我是e同学");
var fstudent=new studentObj("我是f同学");
var gstudent=new studentObj("我是g同学");
var hstudent=new studentObj("我是h同学");
var istudent=new studentObj("我是i同学"); var one = new compositeObj("一班");
var oneOne = new compositeObj("一班一组");
oneOne.add(astudent).add(bstudent);
var oneTwo = new compositeObj("一班二组");
oneTwo.add(cstudent).add(dstudent); one.add(oneOne).add(oneTwo);
var two = new compositeObj("二班");
var twoOne = new compositeObj("二班一组");
twoOne.add(estudent).add(fstudent);
var twoTwo = new compositeObj("二班二组");
twoTwo.add(gstudent).add(hstudent).add(istudent)
two.add(twoOne).add(twoTwo);
var usPcat = new compositeObj("组合设计模式培训学校");
usPcat.add(one).add(two);

(5)客户端调用API,只需要简单的安排去上课即可,也就是客户端只需要写去上课的代码即可


  usPcat.goToClass();
document.write("-------------------------<br>");
usPcat.goToClass("一班");
document.write("-------------------------<br>");
usPcat.goToClass("二班一组");

总结,

 

JavaScript组合设模式--改进上述引入的例子的更多相关文章

  1. JavaScript组合模式---引入

    首先:使用一个例子来引入组合模式,需求为(1)有一个学校有2个班(一班,二班)(2)每个班级分2个小组(一班一组,一班二组,二班一组,二班二组)(3)学校计算机教室有限,每一个小组分着来上课 然后:根 ...

  2. 深入理解JavaScript中创建对象模式的演变(原型)

    深入理解JavaScript中创建对象模式的演变(原型) 创建对象的模式多种多样,但是各种模式又有怎样的利弊呢?有没有一种最为完美的模式呢?下面我将就以下几个方面来分析创建对象的几种模式: Objec ...

  3. JavaScript高级---门面模式设计

    门面模式 两个作用: 1.简化类的接口 2.消除类与使用它的客户代码之间的耦合 门面模式常常是开发人员最亲密的朋友.它几乎是所有javascript库的核心原则 门面模式的目的是为了让开发人员用更简单 ...

  4. javascript闭包(Module模式)的用途和高级使用方式

    javascript闭包(Module模式)的用途和高级使用方式 javascript闭包的用途:1. 匿名自执行函数:或者可以理解为,避免污染全局变量2. 缓存:源于闭包的核心特性便是保存状态,应用 ...

  5. 谈JavaScript组合拼接字符串的效率 --转载

    JavaScript组合拼接字符串的效率.在脚本开发过程中,经常会按照某个规则,组合拼接出一个大字符串进行输出.比如写脚本控件时控制整个控件的外观的HTML标签输出,比如AJAX里得到服务器端的回传值 ...

  6. JavaScript高级---桥模式设计

    <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01//EN" "http://www.w3.org/TR/html4/stri ...

  7. JavaScript组合继承的一点思考

    今天看<JavaScript高级程序设计>一书中关于组合继承模式时.书上有这么一个Demo程序: <html> <head> </head> <b ...

  8. javascript的继承模式

    在javascript里面看到javascript的继承模式和传统的继承模式是有区别的,就想查资料看一下到底有区别,就看到了这篇文章,觉得讲得还可以,暂时先放上来,以后有别的东西再补充: http:/ ...

  9. Java 实现组合(Composite)模式

    类图 /** * 树 总体 * * @author stone * */ public class Tree { private TreeNode root; //根节点 public Tree(St ...

随机推荐

  1. Linux嵌入式 -- 内核 - 内存管理

    1.  逻辑地址 线性地址 物理地址 段式管理: 16位CPU,20根地址总线,可寻址1M内存,但是只有16位的寄存器,64K. 逻辑地址  =  段基地址 + 段内偏移地址 物理地址 PA  = 段 ...

  2. Spring初学之bean之间的关系和bean的作用域

    一.bean之间的关系 <?xml version="1.0" encoding="UTF-8"?> <beans xmlns="h ...

  3. POJ2741 Colored Cubes

    Description There are several colored cubes. All of them are of the same size but they may be colore ...

  4. Don't add unneeded context不要加不需要的文本

  5. 阿里云 linux 找回mysql root密码

    不小心手贱修改了密码,而且使用phpMyAdmin这种自动生成密码,又没记录密码,真实醉了   搜了半天,问题多多,想过回滚磁盘到昨天,在阿里云已经买了付费找密码 最后终于自己解决了,其实很简单 cd ...

  6. java大数字操作:BigInteger,BigDecimal(浮点型)

    java大数字操作: BigInteger:大数字整型的 BigDecimal(浮点型):大数字小数的,也适用大的整数 BigInteger: String num1 = "10038182 ...

  7. DecimalFormat数据格式函数

    DecimalFormat数据格式函数 class FormatDemo2{ public void format(String pattern, double value) { DecimalFor ...

  8. 如何做好App的测试工作

    记得刚开始接触app测试时,可谓是一脸懵状,拿到一个功能不知道该测些什么,会因为测试范围确认不足.测试点考虑不全等导致线上问题,吃一堑才会长一智,栽过几次坑后就学会了如何避免.现总结App测试点如下, ...

  9. 一道问题引出的python中可变数据类型与不可变数据类型

    一. 问题的提出 我们先来看两个对比 第一道题,当对象为整数时,最终结果:b = 2, a = 1,b的变化没有引起a的变化 a = 1 b = a b += 1 print(a) print(b) ...

  10. uva 11088 暴力枚举子集/状压dp

    https://vjudge.net/problem/UVA-11088 对于每一种子集的情况暴力枚举最后一个三人小组取最大的一种情况即可,我提前把三个人的子集情况给筛出来了. 即 f[S]=MAX{ ...