js之组合模式

在命令模式学习完的基础上:

更强大的宏命令

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
    <button id="button">按我</button>
    <script>
        var MacroCommand = function(){
            return {
                commandsList: [],
                add: function( command ){
                    this.commandsList.push( command );
                },
                execute: function(){
                    for ( var i = 0, command; command = this.commandsList[ i++ ]; ){
                        command.execute();
                    }
                }
            }
        };
        var openAcCommand = {
            execute: function(){
                console.log( '打开空调' );
            }
        };
        /**********家里的电视和音响是连接在一起的,所以可以用一个宏命令来组合打开电视和打开音响的命令
         *********/
        var openTvCommand = {
            execute: function(){
                console.log( '打开电视' );
            }
        };
        var openSoundCommand = {
            execute: function(){
                console.log( '打开音响' );
            }
        };
        var macroCommand1 = MacroCommand();
        macroCommand1.add( openTvCommand );
        macroCommand1.add( openSoundCommand );
        /*********关门、打开电脑和打登录 QQ 的命令****************/
        var closeDoorCommand = {
            execute: function(){
                console.log( '关门' );
            }
        };
        var openPcCommand = {
            execute: function(){
                console.log( '开电脑' );
            }
        };
        var openQQCommand = {
            execute: function(){
                console.log( '登录 QQ' );
            }
        };
        var macroCommand2 = MacroCommand();
        macroCommand2.add( closeDoorCommand );
        macroCommand2.add( openPcCommand );
        macroCommand2.add( openQQCommand );
        /*********现在把所有的命令组合成一个“超级命令”**********/
        var macroCommand = MacroCommand();
        macroCommand.add( openAcCommand );
        macroCommand.add( macroCommand1 );
        macroCommand.add( macroCommand2 );
        /*********最后给遥控器绑定“超级命令”**********/
        var setCommand = (function( command ){
            document.getElementById( 'button' ).onclick = function(){
                command.execute();
            }
        })( macroCommand );
    </script>

</body>
</html>

从这个例子中可以看到,基本对象可以被组合成更复杂的组合对象,组合对象又可以被组合,这样不断递归下去,这棵树的结构可以支持任意多的复杂度。在树最终被构造完成之后,让整颗
树最终运转起来的步骤非常简单,只需要调用最上层对象的 execute 方法。每当对最上层的对象进行一次请求时,实际上是在对整个树进行深度优先的搜索,而创建组合对象的程序员并不关心这些内在的细节,往这棵树里面添加一些新的节点对象是非常容易的事情。


组合模式的例子——扫描文件夹

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
    <button id="button">按我</button>
    <script>
        /******************************* Folder ******************************/
        var Folder = function( name ){
            this.name = name;
            this.files = [];
        };
        Folder.prototype.add = function( file ){
            this.files.push( file );
        };
        Folder.prototype.scan = function(){
            console.log( '开始扫描文件夹: ' + this.name );
            for ( var i = 0, file, files = this.files; file = files[ i++ ]; ){
                file.scan();
            }
        };
        /******************************* File ******************************/
        var File = function( name ){
            this.name = name;
        };
        File.prototype.add = function(){
            throw new Error( '文件下面不能再添加文件' );
        };
        File.prototype.scan = function(){
            console.log( '开始扫描文件: ' + this.name );
        };

        var folder = new Folder( '学习资料' );
        var folder1 = new Folder( 'JavaScript' );
        var folder2 = new Folder ( 'jQuery' );

        var file1 = new File( 'JavaScript 设计模式与开发实践' );
        var file2 = new File( '精通 jQuery' );
        var file3 = new File( '重构与模式' )
        folder1.add( file1 );
        folder2.add( file2 );
        folder.add( folder1 );
        folder.add( folder2 );
        folder.add( file3 );
        folder.scan();
    </script>

</body>
</html>

一些值得注意的地方

  1. 组合模式不是父子关系
  2. 对叶对象操作的一致性

level02:引用父对象

在 上个例子中,组合对象保存了它下面的子节点的引用,这是组合模式的特点,此时树结构是从上至下的。但有时候我们需要在子节点上保持对父节点的引用,比如在组合模式中使用职责链时,有可能需要让请求从子节点往父节点上冒泡传递。还有当我们删除某个文件的时候,实际上是从这个文件所在的上层文件夹中删除该文件的。

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
    <script>
        var Folder = function( name ){
            this.name = name;
            this.parent = null; //增加 this.parent 属性
            this.files = [];
        };
        Folder.prototype.add = function( file ){
            file.parent = this; //设置父对象
            this.files.push( file );
        };
        Folder.prototype.scan = function(){
            console.log( '开始扫描文件夹: ' + this.name );
            for ( var i = 0, file, files = this.files; file = files[ i++ ]; ){
                file.scan();
            }
        };
        Folder.prototype.remove = function(){
            if ( !this.parent ){ //根节点或者树外的游离节点
                return;
            }
            for ( var files = this.parent.files, l = files.length - 1; l >=0; l-- ){
                var file = files[ l ];
                if ( file === this ){
                    files.splice( l, 1 );
                }
            }
        };
        var File = function( name ){
            this.name = name;
            this.parent = null;
        };
        File.prototype.add = function(){
            throw new Error( '不能添加在文件下面' );
        };
        File.prototype.scan = function(){
            console.log( '开始扫描文件: ' + this.name );
        };
        File.prototype.remove = function(){
            if ( !this.parent ){ //根节点或者树外的游离节点
                return;
            }
            for ( var files = this.parent.files, l = files.length - 1; l >=0; l-- ){
                var file = files[ l ];
                if ( file === this ){
                    files.splice( l, 1 );
                }
            }
        };
        var folder = new Folder( '学习资料' );
        var folder1 = new Folder( 'JavaScript' );
        var file1 = new Folder ( '深入浅出 Node.js' );
        folder1.add( new File( 'JavaScript 设计模式与开发实践' ) );
        folder.add( folder1 );
        folder.add( file1 );
        file1.remove();
        folder1.remove(); //移除文件夹
        folder.scan();
    </script>
</body>
</html>

何时使用组合模式

组合模式如果运用得当,可以大大简化客户的代码。一般来说,组合模式适用于以下这两种情况。
 表示对象的部分整体层次结构。组合模式可以方便地构造一棵树来表示对象的部分整体结构。特别是我们在开发期间不确定这棵树到底存在多少层次的时候。在树的构造最终完成之后,只需要通过请求树的最顶层对象,便能对整棵树做统一的操作。在组合模式中增加和删除树的节点非常方便,并且符合开放封闭原则。

 客户希望统一对待树中的所有对象。组合模式使客户可以忽略组合对象和叶对象的区别,客户在面对这棵树的时候,不用关心当前正在处理的对象是组合对象还是叶对象,也就不用写一堆 if 、 else 语句来分别处理它们。组合对象和叶对象会各自做自己正确的事情,这是组合模式最重要的能力。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章