成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

優(yōu)化你的執(zhí)行隊列-組合模式

fish / 1418人閱讀

摘要:如果把設(shè)計模式當(dāng)做一門語言的話,我覺得組合模式就是里面的數(shù)組,或者更確切的說,應(yīng)該算是一棵樹。給向上的綁定向上的執(zhí)行程序可以看出,使用組合模式命令模式可以完美的打出個兩個王的終極炸彈。將同類操作同目的操作放在一組。

如果把設(shè)計模式當(dāng)做一門語言的話,我覺得組合模式就是里面的數(shù)組,或者更確切的說,應(yīng)該算是一棵樹。 樹的枝干就是其他模式的使用,比如命令模式,代理模式,單例模式等等??偟膩碚f,組合模式在這里就相當(dāng)于一個容器,但也并非僅僅是個容器(不然,我還給他冠名"模式").

組合模式+命令模式

上篇的命令模式大家應(yīng)該可以了解到,一個命令和命令執(zhí)行者的相關(guān)系數(shù)為 zero . 所以你對命令的執(zhí)行者做什么都不要緊,但是他的接口必須保持一致。 而這個特性映射出一個道理,就是既然我可以do anything, 意味著我可以在執(zhí)行者上面加上其他的執(zhí)行者.
are u ok?
恩恩,I"m fine, fuck u.
當(dāng)然,這時候這個執(zhí)行者就不叫做命令,而應(yīng)該改口叫做宏。來,和上一篇一樣,說球的問題。

//當(dāng)點擊一個球后,讓球往上走,再往右走
var setCommand = function(ele, command) {
    ele.onclick = function() {
        command.exe();
    }
}
var locat = (function() { //執(zhí)行事件類
    var ball = getEle("#pic"),
        style = ball.style; //緩存style
    var move = function(direct) {
        return function() {
            var dir;
            switch (direct) {
                case "bottom":
                    dir = parseInt(getLocat(ball).top) || 0;
                    style.top = `${dir+10}px`;
                    break;
                case "right":
                    dir = parseInt(getLocat(ball).left) || 0;
                    style.left = `${dir+10}px`;
                    break;
                default:
                    dir = parseInt(getLocat(ball)[direct]) || 0;
                    style[direct] = `${dir-10}px`;
            }
        }
    }
    return {
        moveUp: move("top"),
        moveDown: move("bottom"),
        moveLeft: move("left"),
        moveRight: move("right")
    }
})();
//封裝命令
var moveUR = (function(){
    var exe = function(){
        locat.moveUp();
        locat.moveRight();
    }
    return  {  //封裝命令
        exe
    }
})();
setCommand(getEle("leftUpBtn"), moveUR); //綁定執(zhí)行效果的程序

可以看出,上面的代碼可以完成基本任務(wù)。但是,如果某一天你的leader要求,讓小球 上->右->下->左 進(jìn)行一個循環(huán)回到原始的位置,恐怕你的反應(yīng)不是很強烈。就是在moveUR里面再加些東西唄。呵呵,來,你leader又說了。在上->右->下的時候轉(zhuǎn)個圈然后->左.(Ps: 轉(zhuǎn)個毛線啊~~~~), 表示聽到這里,想想某天你的leader抽風(fēng)了,又換一個實現(xiàn)方式。 我想,你年都過不好~。
所以,為了讓你早些回家,早點見爹媽,早點領(lǐng)年終獎,早點買春運火車票~
我相信,命令模式是你不二之選.
我們使用命令模式重構(gòu)一下.

//上面部分不動,只將下面部分改寫
//命令集合
var command = function(){
    var lists = [];
    return {
        add(command){
            lists.push(command);
        },
        exe(){
            for(var i = 0,command;command = lists[i++];){
                command.exe();
            }
        }
    }
}
//封裝命令
var moveUp = (function(){
    var exe = function(){
        locat.moveUp();
    }
    return {
        exe
    }
})();
var moveLeft = (function(){
    var exe = function(){
        locat.moveLeft();
    }
    return{
        exe
    }
})();
var moveGroup1 = command();  //得到一個命令集
moveGroup1.add(moveUp);  //添加向上的命令
moveGroup1.add(moveLeft);  //。。。。
setCommand(getEle("leftUpBtn"), moveGroup1); //給向上的button,綁定向上的執(zhí)行程序

可以看出,使用組合模式+命令模式可以完美的打出4個2+兩個王的終極炸彈。 你可以隨機的添加你想要的效果,但是前提還是必須保證有一致的接口內(nèi)容。
現(xiàn)在可以看出組合模式的好處了吧。 想想,這個模式的特點--隊列。沒錯,組合模式會創(chuàng)建一棵樹,而這棵樹的枝干全部是有你來添加,你可以讓他變成百年梧桐,也可以讓他變成行道樹,而且調(diào)用的方式極其簡單,使用一個接口,這棵樹會把你的命令通過莖,一個接一個的輸送過去。

talk is cheap, show u code

var moveUP = (function(){
    var exe = funciton(){
        conosle.log("moveUP");
    }
    return {
        exe
    }
})();
var moveLeft = (function(){
    var exe = funciton(){
        conosle.log("moveLeft");
    }
    return {
        exe
    }
})();
var moveBottom = (function(){
    var exe = funciton(){
        conosle.log("moveBottom");
    }
    return {
        exe
    }
})();
var moveRight = (function(){
    var exe = funciton(){
        conosle.log("moveRight");
    }
    return {
        exe
    }
})();
//加一點難度,對命令進(jìn)行分組
var moveGrop1 = command();
moveGrop1.add(moveUp);   //moveUp
moveGrop1.add(moveLeft);  //moveLeft
var moveGrop2 = command();
moveGrop2.add(moveBottom);  //moveBottom
moveGrop2.add(moveRight);  //moveRight
var moveGrop3 = command();
moveGrop3.add(moveBottom);  //moveBottom
moveGrop3.add(moveBottom);  //moveBottom
var moveGrop = command();
moveGrop.add(moveGrop1); 
moveGrop.add(moveGrop2);
moveGrop.add(moveGrop3);
moveGrop.exe();  //接下來會按上面的順序一個一個的運行。

可以看出,組合模式最大的一個特點就是分組操作。將同類操作(同目的操作)放在一組。 就像做一個gif了,一個幀+一個幀+一個幀... 比如,我可以蹲下來,然后起跳。 我又可以蹲下來,然后站立,走路。 就可以把一個片段,一個片段組合成你最想要的效果(懷念高中時候沒有好好學(xué)習(xí)排列組合 :(] 而且重用性也是杠杠的。
可以看出,上面的add這樣寫好蠢。。。 為了滿足裝逼的需求,決定優(yōu)化一下。

//命令集合
var command = function(){
  var lists = [];
    return {
        add(){
            for(var i = 0,command; command = arguments[i++];){
                lists.push(command);
            }
          console.log(lists);
        },
        exe(){
            for(var i = 0,command;command = lists[i++];){
                command.exe();
            }
        }
    }
}
//添加命令
var moveGrop1 = command();
moveGrop1.add(moveUp,moveLeft);   //moveUp,moveLeft
var moveGrop2 = command();
moveGrop2.add(moveBottom,moveRight);  //moveBottom,moveRight
var moveGrop3 = command();
moveGrop3.add(moveBottom,moveBottom);  //moveBottom,moveBottom
var moveGrop = command();
moveGrop.add(moveGrop1,moveGrop2,moveGrop3); 
moveGrop.exe();  //接下來會按上面的順序一個一個的運行。

干凈,漂亮,美美的.
但是,世上沒有完美的代碼,命令模式來擴展宏命令,這個是他的劍,也是他的死穴。 這樣會造成,你的根節(jié)點和子節(jié)點(不可擴展的節(jié)點)之間層次的不分明。所以,一般,我們會在子節(jié)點上面加上說明,防止意外添加而導(dǎo)致的出錯,不然,bug一出,春運票你也別想買了。
舉個栗子:

var moveGrop = command();
moveGrop1.add(moveUP);
moveUP.add(moveLeft);

雖然看上去這段代碼很蠢,但事實上,我確實踩過===我也很蠢xxx... 所以,這里希望警戒大家,希望把這個錯誤的發(fā)生率降到 0.000000001%.吧。
當(dāng)然,一個藝術(shù)家,往往會給自己留一手。

var moveUp = (function(){
    var exe = funciton(){
        conosle.log("moveUp");
    }
    var add = function(){
        throw "你很蠢,不,你真的很蠢."
    }
    return {
        exe,add
    }
})();

這樣,萬一哪天,真的蠢了,好歹也知道自己怎么蠢的。自己知道自己蠢就可以了,記得代碼寫完后,把這段給刪了(紅臉).

組合模式注意tips

組合模式雖然很strong.但是有些概念性問題,我們還是必須分清楚.

組合模式里,根節(jié)點和子節(jié)點并不是父子關(guān)系。即,他們之間,并不存在繼承關(guān)系。只是他們接口名一致而已。

字節(jié)點對象必須是同類的,這里說說的同類是指完成同一個目的相互協(xié)作的。如果將另外一個組的子節(jié)點插進(jìn)來,造成的結(jié)果就是over.

盡量給根節(jié)點和子節(jié)點取望文生義的名字,后面最好標(biāo)注他們的屬性(根,子).

組合模式的其他用處

這個用處出自于< AlloyTeam的曾探>.(一枚我超級崇拜的大神)
組合模式的特點是子對象地位平等,特別符合我們所說的文件掃描功能。 即,文件掃描只需要知道你的文件名,以及所處的位置即可。
這里我們可以把文件比作子節(jié)點,文件夾比作父節(jié)點.

var folder = function(name){
    return {
        name,  //文件夾名
        files:[],
        add(){
            for(var i = 0,file;file = arguments[i++];){
                file.parent = this;
                this.files.push(file);
            }
        },
        scan(){  //掃描文件
            for(var file of this.files){
                console.log(file.name);  
            }
        },
        remove(){
            if(!this.parent){
                alert("該文件是更目錄,不能刪除!");
                return;
            }
            for(var files = this.parent.files,len = files.length-1;len>=0;len--){  //倒敘遍歷文件夾
                var file = files[len];
                if(file === this){
                    files.splice(len,1);  //刪除文件
                    break;
                }
            }
        }
    }
}
var file = function(name){
    return {
        name,
        scan(){
            console.log(`this is ${name}`);
        },
        remove(){
             if(!this.parent){
                alert("該文件是更目錄,不能刪除!");
                return;
            }
            for(var files = this.parent.files,len = files.length-1;len>=0;len--){  //倒敘遍歷文件夾
                var file = files[len];
                if(file === this){

                    files.splice(len,1);  //刪除文件
                    break;
                }
            }
        },
        add(){
            throw "你很蠢,不,你真的很蠢!"
        }
    }
}
var jimmyFolder = folder("jimmy");
var avi1 = file("小澤瑪利亞.avi");
var avi2 = file("上野真知子.avi");
jimmyFolder.add(avi1,avi2);
jimmyFolder.scan();  //小澤瑪利亞.avi , 上野真知子.avi
avi1.remove();
jimmyFolder.scan();  //上野真知子.avis

恩,get(工口).
當(dāng)然,這只是一個比較簡單的例子,隨著你對根節(jié)點的不斷操作,后面遇到的問題,肯定會凸顯出來,要知道,這樣,組合模式里面的節(jié)點保存是非常耗內(nèi)存的,所以這個模式肯定還有很多可以優(yōu)化的地方.
最后,還是那句話,不要為了模式而模式。 對于模式而言還是希望先寫,后添加。
ending~

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/78505.html

相關(guān)文章

  • 金三銀四背后,一個 Android 程序員的面試心得

    摘要:到十二月份,公司開始第二波裁員,我決定主動拿賠償走人。加一個小插曲上面的題是餓了嗎面試問到的。想去的公司沒有面試好,不要氣餒,繼續(xù)加油準(zhǔn)備。避免打擊自信心。 回顧一下自己這段時間的經(jīng)歷,九月份的時候,公司通知了裁員,我匆匆忙忙地出去面了幾家,但最終都沒有拿到offer,我感覺今年的寒冬有點冷。到十二月份,公司開始第二波裁員,我決定主動拿賠償走人。后續(xù)的面試過程我做了一些準(zhǔn)備,基本都能走...

    Achilles 評論0 收藏0
  • 如何讓你的JavaScript代碼更加語義化

    摘要:二提高代碼語義性針對上述三個案例,用更加語義化的方式來呈現(xiàn)代碼語義化變量類型判斷我覺得不需要太多的解釋,對比顯得清新多了吧。 語義化這個詞在 HTML 中用的比較多,即根據(jù)內(nèi)容的結(jié)構(gòu)化選擇合適的標(biāo)簽。其作用不容小覷: 賦予標(biāo)簽含義,讓代碼結(jié)構(gòu)更加清晰,雖然我們可以在標(biāo)簽上添加 class 來標(biāo)識,但這種通過屬性來表示本體的形式會顯得不夠直接,而且在一定程度上也有冗余。 優(yōu)化搜索引擎...

    Jaden 評論0 收藏0
  • 精讀《Scheduling in React》

    摘要:調(diào)度系統(tǒng),支持不同渲染優(yōu)先級,對進(jìn)行調(diào)度。調(diào)度帶來的限制調(diào)度系統(tǒng)也存在兩個問題。調(diào)度系統(tǒng)能力有限,只能在瀏覽器提供的能力范圍內(nèi)進(jìn)行調(diào)度,而無法影響比如的渲染回收周期。精讀關(guān)于調(diào)度系統(tǒng)的剖析,可以讀深入剖析這篇文章,感謝我們團(tuán)隊的淡蒼提供。 1. 引言 這次介紹的文章是 scheduling-in-react,簡單來說就是 React 的調(diào)度系統(tǒng),為了得到更順滑的用戶體驗。 畢竟前端做到...

    LeexMuller 評論0 收藏0
  • Java面試 32個核心必考點完全解析

    摘要:如問到是否使用某框架,實際是是問該框架的使用場景,有什么特點,和同類可框架對比一系列的問題。這兩個方向的區(qū)分點在于工作方向的側(cè)重點不同。 [TOC] 這是一份來自嗶哩嗶哩的Java面試Java面試 32個核心必考點完全解析(完) 課程預(yù)習(xí) 1.1 課程內(nèi)容分為三個模塊 基礎(chǔ)模塊: 技術(shù)崗位與面試 計算機基礎(chǔ) JVM原理 多線程 設(shè)計模式 數(shù)據(jù)結(jié)構(gòu)與算法 應(yīng)用模塊: 常用工具集 ...

    JiaXinYi 評論0 收藏0

發(fā)表評論

0條評論

fish

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<