設(shè)計(jì)模式系列 — 命令模式

點(diǎn)贊再看,養(yǎng)成習(xí)慣肮疗,公眾號搜一搜【一角錢技術(shù)】關(guān)注更多原創(chuàng)技術(shù)文章额划。本文 GitHub org_hejianhui/JavaStudy 已收錄,有我的系列文章。

前言

23種設(shè)計(jì)模式快速記憶的請看上面第一篇效拭,本篇和大家一起來學(xué)習(xí)命令模式相關(guān)內(nèi)容。

模式定義

將一個(gè)請求封裝為一個(gè)對象胖秒,使發(fā)出請求的責(zé)任和執(zhí)行請求的責(zé)任分割開缎患。這樣兩者之間通過命令對象進(jìn)行溝通,這樣方便將命令對象進(jìn)行儲存阎肝、傳遞挤渔、調(diào)用、增加與管理风题。

在軟件開發(fā)系統(tǒng)中判导,常常出現(xiàn)“方法的請求者”與“方法的實(shí)現(xiàn)者”之間存在緊密的耦合關(guān)系。這不利于軟件功能的擴(kuò)展與維護(hù)沛硅。例如眼刃,想對行為進(jìn)行“撤銷、重做摇肌、記錄”等處理都很不方便擂红,因此“如何將方法的請求者與方法的實(shí)現(xiàn)者解耦?”變得很重要围小,命令模式能很好地解決這個(gè)問題昵骤。

模版實(shí)現(xiàn)如下

package com.niuh.designpattern.command.v1;

/**
 * <p>
 * 命令模式
 * </p>
 */
public class CommandPattern {
    public static void main(String[] args) {
        Command cmd = new ConcreteCommand();
        Invoker ir = new Invoker(cmd);
        System.out.println("客戶訪問調(diào)用者的call()方法...");
        ir.call();
    }
}

//抽象命令
interface Command {
    public abstract void execute();
}

//具體命令
class ConcreteCommand implements Command {
    private Receiver receiver;

    ConcreteCommand() {
        receiver = new Receiver();
    }

    public void execute() {
        receiver.action();
    }
}

//接收者
class Receiver {
    public void action() {
        System.out.println("接收者的action()方法被調(diào)用...");
    }
}

//調(diào)用者
class Invoker {
    private Command command;

    public Invoker(Command command) {
        this.command = command;
    }

    public void setCommand(Command command) {
        this.command = command;
    }

    public void call() {
        System.out.println("調(diào)用者執(zhí)行命令command...");
        command.execute();
    }
}

輸出結(jié)果如下

客戶訪問調(diào)用者的call()方法...
調(diào)用者執(zhí)行命令command...
接收者的action()方法被調(diào)用...

解決的問題

在軟件系統(tǒng)中树碱,行為請求者與行為實(shí)現(xiàn)者通常是一種緊耦合的關(guān)系,但某些場合涉茧,比如需要對行為進(jìn)行記錄赴恨、撤銷或重做、事務(wù)等處理時(shí)伴栓,這種無法抵御變化的緊耦合的設(shè)計(jì)就不太合適伦连。

模式組成

可以將系統(tǒng)中的相關(guān)操作抽象成命令,使調(diào)用者與實(shí)現(xiàn)者相關(guān)分離钳垮,其結(jié)構(gòu)如下惑淳。

組成(角色) 作用
抽象命令類(Command)角色 聲明執(zhí)行命令的接口,擁有執(zhí)行命令的抽象方法 execute()饺窿。
具體命令角色(Concrete Command)角色 是抽象命令類的具體實(shí)現(xiàn)類歧焦,它擁有接收者對象,并通過調(diào)用接收者的功能來完成命令要執(zhí)行的操作肚医。
實(shí)現(xiàn)者/接收者(Receiver)角色 執(zhí)行命令功能的相關(guān)操作绢馍,是具體命令對象業(yè)務(wù)的真正實(shí)現(xiàn)者。
調(diào)用者/請求者(Invoker)角色 是請求的發(fā)送者肠套,它通常擁有很多的命令對象舰涌,并通過訪問命令對象來執(zhí)行相關(guān)請求,它不直接訪問接收者你稚。

實(shí)例說明

實(shí)例概況

結(jié)合命令模式瓷耙,實(shí)現(xiàn)一個(gè)課程視頻的打開和關(guān)閉。


使用步驟

步驟1:聲明執(zhí)行命令的接口刁赖,擁有執(zhí)行命令的抽象方法 execute()

interface Command {
    void execute();
}

步驟2:定義具體命令角色搁痛,創(chuàng)建打開課程鏈接 和 關(guān)閉課程連接

/**
 * 打開課程鏈接
 */
class OpenCourseVideoCommand implements Command {

    private CourseVideo courseVideo;

    public OpenCourseVideoCommand(CourseVideo courseVideo) {
        this.courseVideo = courseVideo;
    }

    @Override
    public void execute() {
        courseVideo.open();
    }
}

/**
 * 關(guān)閉課程鏈接
 */
class CloseCourseVideoCommand implements Command {

    private CourseVideo courseVideo;

    public CloseCourseVideoCommand(CourseVideo courseVideo) {
        this.courseVideo = courseVideo;
    }

    @Override
    public void execute() {
        courseVideo.close();
    }
}

步驟3:定義接收者角色,執(zhí)行命令功能的相關(guān)操作宇弛,是具體命令對象業(yè)務(wù)的真正實(shí)現(xiàn)者

class CourseVideo {

    private String name;

    public CourseVideo(String name) {
        this.name = name;
    }

    public void open() {
        System.out.println(this.name + "課程視頻開放鸡典。");
    }

    public void close() {
        System.out.println(this.name + "課程視頻關(guān)閉。");
    }
}

步驟4:創(chuàng)建User對象為請求的發(fā)送者枪芒,即請求者角色

class User {

    private List<Command> commands = new ArrayList<>();

    public void addCommand(Command command) {
        commands.add(command);
    }

    public void executeCommands() {
        commands.forEach(Command::execute);
        commands.clear();
    }
}

步驟4:測試執(zhí)行

public class CommandPattern {

    public static void main(String[] args) {
        //命令接收者
        CourseVideo courseVideo = new CourseVideo("設(shè)計(jì)模式系列");

        //創(chuàng)建命令
        OpenCourseVideoCommand openCourseVideoCommand = new OpenCourseVideoCommand(courseVideo);
        CloseCourseVideoCommand closeCourseVideoCommand = new CloseCourseVideoCommand(courseVideo);

        //創(chuàng)建執(zhí)行人
        User user = new User();

        //添加命令
        user.addCommand(openCourseVideoCommand);
        user.addCommand(closeCourseVideoCommand);

        //執(zhí)行
        user.executeCommands();
    }
}

輸出結(jié)果

設(shè)計(jì)模式系列課程視頻開放彻况。
設(shè)計(jì)模式系列課程視頻關(guān)閉。

優(yōu)點(diǎn)

  1. 降低系統(tǒng)的耦合度病苗。命令模式能將調(diào)用操作的對象與實(shí)現(xiàn)該操作的對象解耦疗垛。
  2. 增加或刪除命令非常方便。采用命令模式增加與刪除命令不會影響其他類硫朦,它滿足“開閉原則”贷腕,對擴(kuò)展比較靈活。
  3. 可以實(shí)現(xiàn)宏命令。命令模式可以與組合模式結(jié)合泽裳,將多個(gè)命令裝配成一個(gè)組合命令瞒斩,即宏命令。
  4. 方便實(shí)現(xiàn) Undo 和 Redo 操作涮总。命令模式可以與后面介紹的備忘錄模式結(jié)合胸囱,實(shí)現(xiàn)命令的撤銷與恢復(fù)。

缺點(diǎn)

可能產(chǎn)生大量具體命令類瀑梗。因?yàn)橛?jì)對每一個(gè)具體操作都需要設(shè)計(jì)一個(gè)具體命令類烹笔,這將增加系統(tǒng)的復(fù)雜性。

應(yīng)用場景

命令執(zhí)行過程較為復(fù)雜且可能存在變化抛丽,需要對執(zhí)行命令動(dòng)作本身進(jìn)行額外操作谤职,此時(shí)可以考慮使用命令模式

命令模式的擴(kuò)展

在軟件開發(fā)中,有時(shí)將命令模式與組合模式聯(lián)合使用亿鲜,這就構(gòu)成了宏命令模式允蜈,也叫組合命令模式。宏命令包含了一組命令蒿柳,它充當(dāng)了具體命令與調(diào)用者的雙重角色饶套,執(zhí)行它時(shí)將遞歸調(diào)用它所包含的所有命令,其具體結(jié)構(gòu)圖如下:


模版實(shí)現(xiàn)如下

package com.niuh.designpattern.command.v2;

import java.util.ArrayList;

/**
 * <p>
 * 組合命令模式
 * </p>
 */
public class CompositeCommandPattern {
    public static void main(String[] args) {
        AbstractCommand cmd1 = new ConcreteCommand1();
        AbstractCommand cmd2 = new ConcreteCommand2();
        CompositeInvoker ir = new CompositeInvoker();
        ir.add(cmd1);
        ir.add(cmd2);
        System.out.println("客戶訪問調(diào)用者的execute()方法...");
        ir.execute();
    }
}

//抽象命令
interface AbstractCommand {
    public abstract void execute();
}

//樹葉構(gòu)件: 具體命令1
class ConcreteCommand1 implements AbstractCommand {
    private CompositeReceiver receiver;

    ConcreteCommand1() {
        receiver = new CompositeReceiver();
    }

    public void execute() {
        receiver.action1();
    }
}

//樹葉構(gòu)件: 具體命令2
class ConcreteCommand2 implements AbstractCommand {
    private CompositeReceiver receiver;

    ConcreteCommand2() {
        receiver = new CompositeReceiver();
    }

    public void execute() {
        receiver.action2();
    }
}

//樹枝構(gòu)件: 調(diào)用者
class CompositeInvoker implements AbstractCommand {
    private ArrayList<AbstractCommand> children = new ArrayList<AbstractCommand>();

    public void add(AbstractCommand c) {
        children.add(c);
    }

    public void remove(AbstractCommand c) {
        children.remove(c);
    }

    public AbstractCommand getChild(int i) {
        return children.get(i);
    }

    public void execute() {
        for (Object obj : children) {
            ((AbstractCommand) obj).execute();
        }
    }
}

//接收者
class CompositeReceiver {
    public void action1() {
        System.out.println("接收者的action1()方法被調(diào)用...");
    }

    public void action2() {
        System.out.println("接收者的action2()方法被調(diào)用...");
    }
}

輸出結(jié)果如下

客戶訪問調(diào)用者的execute()方法...
接收者的action1()方法被調(diào)用...
接收者的action2()方法被調(diào)用...

命令模式還可以同備忘錄(Memento)模式組合使用垒探,這樣就變成了可撤銷的命令模式

源碼中的應(yīng)用

在 JdbcTemplate 中的應(yīng)用

在JdbcTemplate中命令模式的使用并沒有遵從標(biāo)準(zhǔn)的命令模式的使用妓蛮,只是思想相同而已。

在 Spring 的 JdbcTemplate 這個(gè)類中有 query() 方法叛复,query() 方法中定義了一個(gè)內(nèi)部類 QueryStatementCallback仔引,QueryStatementCallback 又實(shí)現(xiàn)了 StatementCallback 接口扔仓,另外還有其它類實(shí)現(xiàn)了該接口褐奥,StatementCallback 接口中又有一個(gè)抽象方法 doInStatement()。在 execute() 中又調(diào)用了 query()翘簇。

StatementCallback充當(dāng)?shù)氖敲罱巧?code>JdbcTemplate即充當(dāng)調(diào)用者角色撬码,又充當(dāng)接收者角色。上面的類圖只是為了方便理解版保,實(shí)際上呜笑,QueryStatementCallback 與 ExecuteStatementCallback是JdbcTemplate中方法的內(nèi)部類,具體看源碼中的內(nèi)容彻犁。

部分源碼分析

StatementCallback接口:

public interface StatementCallback<T> {
    T doInStatement(Statement stmt) throws SQLException, DataAccessException;
}

JdbcTemplate類:

public class JdbcTemplate extends JdbcAccessor implements JdbcOperations {
    //相當(dāng)于調(diào)用者發(fā)布的一個(gè)命令
    @Override
    public <T> List<T> query(String sql, RowMapper<T> rowMapper) throws DataAccessException {
        return query(sql, new RowMapperResultSetExtractor<T>(rowMapper));
    }
    //命令發(fā)布后由具體的命令派給接收者進(jìn)行執(zhí)行
    @Override
    public <T> T query(final String sql, final ResultSetExtractor<T> rse) throws DataAccessException {
        Assert.notNull(sql, "SQL must not be null");
        Assert.notNull(rse, "ResultSetExtractor must not be null");
        if (logger.isDebugEnabled()) {
            logger.debug("Executing SQL query [" + sql + "]");
        }
        //內(nèi)部類叫胁,實(shí)現(xiàn)StatementCallback,相當(dāng)于具體的命令
        class QueryStatementCallback implements StatementCallback<T>, SqlProvider {
            @Override
            public T doInStatement(Statement stmt) throws SQLException {
                ResultSet rs = null;
                try {
                    rs = stmt.executeQuery(sql);
                    ResultSet rsToUse = rs;
                    if (nativeJdbcExtractor != null) {
                        rsToUse = nativeJdbcExtractor.getNativeResultSet(rs);
                    }
                    return rse.extractData(rsToUse);
                }
                finally {
                    JdbcUtils.closeResultSet(rs);
                }
            }
            @Override
            public String getSql() {
                return sql;
            }
        }
        return execute(new QueryStatementCallback());
    }
    //相當(dāng)于接收者汞幢,命令真正的執(zhí)行者
    @Override
    public <T> T execute(StatementCallback<T> action) throws DataAccessException {
        Assert.notNull(action, "Callback object must not be null");
        Connection con = DataSourceUtils.getConnection(getDataSource());
        Statement stmt = null;
        try {
            Connection conToUse = con;
            if (this.nativeJdbcExtractor != null &&
                    this.nativeJdbcExtractor.isNativeConnectionNecessaryForNativeStatements()) {
                conToUse = this.nativeJdbcExtractor.getNativeConnection(con);
            }
            stmt = conToUse.createStatement();
            applyStatementSettings(stmt);
            Statement stmtToUse = stmt;
            if (this.nativeJdbcExtractor != null) {
                stmtToUse = this.nativeJdbcExtractor.getNativeStatement(stmt);
            }
            T result = action.doInStatement(stmtToUse);
            handleWarnings(stmt);
            return result;
        }
        catch (SQLException ex) {
            // Release Connection early, to avoid potential connection pool deadlock
            // in the case when the exception translator hasn't been initialized yet.
            JdbcUtils.closeStatement(stmt);
            stmt = null;
            DataSourceUtils.releaseConnection(con, getDataSource());
            con = null;
            throw getExceptionTranslator().translate("StatementCallback", getSql(action), ex);
        }
        finally {
            JdbcUtils.closeStatement(stmt);
            DataSourceUtils.releaseConnection(con, getDataSource());
        }
    }
}

PS:以上代碼提交在 Githubhttps://github.com/Niuh-Study/niuh-designpatterns.git

文章持續(xù)更新驼鹅,可以公眾號搜一搜「 一角錢技術(shù) 」第一時(shí)間閱讀, 本文 GitHub org_hejianhui/JavaStudy 已經(jīng)收錄,歡迎 Star输钩。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末豺型,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子买乃,更是在濱河造成了極大的恐慌姻氨,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,941評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件剪验,死亡現(xiàn)場離奇詭異肴焊,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)功戚,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,397評論 3 395
  • 文/潘曉璐 我一進(jìn)店門抖韩,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人疫铜,你說我怎么就攤上這事茂浮。” “怎么了壳咕?”我有些...
    開封第一講書人閱讀 165,345評論 0 356
  • 文/不壞的土叔 我叫張陵席揽,是天一觀的道長。 經(jīng)常有香客問我谓厘,道長幌羞,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,851評論 1 295
  • 正文 為了忘掉前任竟稳,我火速辦了婚禮属桦,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘他爸。我一直安慰自己聂宾,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,868評論 6 392
  • 文/花漫 我一把揭開白布诊笤。 她就那樣靜靜地躺著系谐,像睡著了一般。 火紅的嫁衣襯著肌膚如雪讨跟。 梳的紋絲不亂的頭發(fā)上纪他,一...
    開封第一講書人閱讀 51,688評論 1 305
  • 那天,我揣著相機(jī)與錄音晾匠,去河邊找鬼茶袒。 笑死,一個(gè)胖子當(dāng)著我的面吹牛凉馆,可吹牛的內(nèi)容都是我干的薪寓。 我是一名探鬼主播乾巧,決...
    沈念sama閱讀 40,414評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼预愤!你這毒婦竟也來了沟于?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,319評論 0 276
  • 序言:老撾萬榮一對情侶失蹤植康,失蹤者是張志新(化名)和其女友劉穎旷太,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體销睁,經(jīng)...
    沈念sama閱讀 45,775評論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡供璧,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,945評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了冻记。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片睡毒。...
    茶點(diǎn)故事閱讀 40,096評論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖冗栗,靈堂內(nèi)的尸體忽然破棺而出演顾,到底是詐尸還是另有隱情,我是刑警寧澤隅居,帶...
    沈念sama閱讀 35,789評論 5 346
  • 正文 年R本政府宣布钠至,位于F島的核電站,受9級特大地震影響胎源,放射性物質(zhì)發(fā)生泄漏棉钧。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,437評論 3 331
  • 文/蒙蒙 一涕蚤、第九天 我趴在偏房一處隱蔽的房頂上張望宪卿。 院中可真熱鬧,春花似錦万栅、人聲如沸佑钾。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,993評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽次绘。三九已至瘪阁,卻和暖如春撒遣,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背管跺。 一陣腳步聲響...
    開封第一講書人閱讀 33,107評論 1 271
  • 我被黑心中介騙來泰國打工义黎, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人豁跑。 一個(gè)月前我還...
    沈念sama閱讀 48,308評論 3 372
  • 正文 我出身青樓廉涕,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子狐蜕,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,037評論 2 355