Java設(shè)計(jì)模式百例 - 模板方法模式

本文源碼見:https://github.com/get-set/get-designpatterns/tree/master/template-method

在模板方法模式(Template Method Pattern)中蝶棋,一個(gè)抽象類公開定義了執(zhí)行它的方法的方式/模板。它的子類可以按需要重寫方法實(shí)現(xiàn)按傅,但調(diào)用將以抽象類中定義的方式進(jìn)行逆粹。這種類型的設(shè)計(jì)模式屬于行為型模式荒揣。

關(guān)于模板怎抛,大家生活中都有體會(huì):

  • 我們總感覺新聞聯(lián)播里的新聞?dòng)行┕潭ǖ摹疤茁贰惫渥辏热?code>______在____的陪同下泞歉,不遠(yuǎn)萬里,來到_____家中宇弛,為_____帶來了節(jié)日的祝福和良好的祝愿鸡典,并饒有興致的觀看了_____。____握著___的手激動(dòng)的說_______枪芒。
  • 我們學(xué)生期間填了各種表格彻况,寫了不少報(bào)告,比如實(shí)驗(yàn)報(bào)告:1舅踪、實(shí)驗(yàn)?zāi)康暮鸵螅?纽甘、實(shí)驗(yàn)設(shè)備(環(huán)境)及要求;3硫朦、實(shí)驗(yàn)步驟贷腕;4、實(shí)驗(yàn)結(jié)果咬展;5泽裳、討論和分析

這樣的例子能舉出好多來破婆,這就是模板涮总,里邊的空或步驟是我們具體要去補(bǔ)充的地方。為什么要搞這些模板出來呢祷舀?規(guī)范瀑梗,為了希望大家都按照一定的規(guī)范約束來實(shí)現(xiàn)。這也是我們?cè)陬愱P(guān)系的設(shè)計(jì)中裳扯,采用這種設(shè)計(jì)模式的初衷抛丽。

例子

下邊這個(gè)例子是通過學(xué)生或老師的自我介紹來演示模板方法模式的應(yīng)用。

無論老師還是學(xué)生饰豺,在做自我介紹時(shí)通常都有自我基本情況介紹亿鲜、獲得過什么獎(jiǎng)勵(lì)、求學(xué)/教學(xué)經(jīng)歷等方面的內(nèi)容冤吨,可以認(rèn)為是一個(gè)模板蒿柳。我們的例子中,自我介紹包含兩部分漩蟆,基本情況介紹和表決心(哈哈)垒探,如下:

SchoolPerson.java

public abstract class SchoolPerson {
    protected String name;
    protected int age;
    protected String schoolName;
    protected String hometown;

    public SchoolPerson(String name, int age, String schoolName, String hometown) {
        this.name = name;
        this.age = age;
        this.schoolName = schoolName;
        this.hometown = hometown;
    }

    public void selfIntroduction() {
        myBasicInfo();
        myslogan();
    }

    public abstract void myBasicInfo();

    public abstract void mySlogan();
}

我們看到,這個(gè)模板如果作為填空題的話怠李,“空”就在myBasicInfomySlogan兩個(gè)方法上圾叼。這兩個(gè)方法是抽象的,要求子類去實(shí)現(xiàn)捺癞。

Student.java

public class Student extends SchoolPerson {
    public Student(String name, int age, String schoolName, String hometown) {
        super(name, age, schoolName, hometown);
    }

    public void myBasicInfo() {
        System.out.println("我是一名學(xué)生夷蚊,名叫" + this.name + ", 今年" + this.age + "歲翘簇, " + this.hometown + "人撬码, 在" + this.schoolName + "上學(xué)。");
    }

    public void mySlogan() {
        System.out.println("在我在" + this.schoolName + "求學(xué)的過程中版保,我一定 好好學(xué)習(xí)呜笑,天天向上!");
    }
}

Teacher.java

public class Teacher extends SchoolPerson {

    public Teacher(String name, int age, String schoolName, String hometown) {
        super(name, age, schoolName, hometown);
    }

    public void myBasicInfo() {
        System.out.println("我是一名教師彻犁,名叫" + this.name + "叫胁, 今年" + this.age + "歲, " + this.hometown + "人汞幢, 在" + this.schoolName + "教書驼鹅。");
    }

    public void mySlogan() {
        System.out.println("在我在" + this.schoolName + "教學(xué)的過程中,我一定 為人師表,誨人不倦输钩!");
    }
}

學(xué)生和老師對(duì)兩個(gè)模板方法都有不同的實(shí)現(xiàn)豺型。我們看一下執(zhí)行效果:

Client.java

public class Client {
    public static void main(String[] args) {
        SchoolPerson student = new Student("張三", 12, "光明小學(xué)", "山東濟(jì)南");
        student.selfIntroduction();

        SchoolPerson teacher = new Teacher("李四", 32, "光明小學(xué)", "山東青島");
        teacher.selfIntroduction();
    }
}

輸出為:

我是一名學(xué)生,名叫張三买乃, 今年12歲姻氨, 山東濟(jì)南人, 在光明小學(xué)上學(xué)剪验。
在我在光明小學(xué)求學(xué)的過程中肴焊,我一定 好好學(xué)習(xí),天天向上功戚!
我是一名教師娶眷,名叫李四, 今年32歲啸臀, 山東青島人届宠, 在光明小學(xué)教書。
在我在光明小學(xué)教學(xué)的過程中壳咕,我一定 為人師表席揽,誨人不倦!

總結(jié)

看這個(gè)例子谓厘,你可能會(huì)感覺幌羞,這就是繼承嘛,有啥新鮮的竟稳?的確属桦,雖然我們以前一直說類的組合(或說委托)優(yōu)先于類的繼承來使用。但是模板方法模式是為數(shù)不多的為我們示范關(guān)于繼承的使用方式的設(shè)計(jì)模式他爸。

模板方法模式的特點(diǎn)很好總結(jié)聂宾,它將一般性的可復(fù)用的行為由基類固化,而把特殊化的行為交由具體的子類來實(shí)現(xiàn)诊笤。具體來說:

  1. 子類通常不關(guān)心全局(比如整個(gè)流程系谐、提綱、步驟)讨跟,而只負(fù)責(zé)”填空“纪他;”填空“通過實(shí)現(xiàn)或重寫父類的方法來實(shí)現(xiàn)。
  2. 從父類角度晾匠,全局性的規(guī)范約束掌握在自己手中茶袒,具體來說通過模板方法來約束,從而能夠盡量簡化子類的復(fù)雜度凉馆。父類并不一定是抽象類薪寓,模板方法也并不一定是抽象方法亡资。

再簡單介紹一個(gè)實(shí)際的例子。我們知道向叉,Servlet有特定的規(guī)范锥腻,以便Servlet的容器(比如tomcat)和Servlet的實(shí)現(xiàn)(我們開發(fā)的JavaEE應(yīng)用)能夠很好的合作。

其中javax.servlet.http.HttpServlet就是Servlet規(guī)范在Http協(xié)議方面的”踐行者“植康。HttpServlet也是一種Servlet旷太,因此也必須有service方法來提供服務(wù)展懈。但是Http有其具體的服務(wù)分類(GET销睁、POST、PUT等不同的請(qǐng)求方法)存崖,我們看一下HttpServlet.service方法的實(shí)現(xiàn):

HttpServlet.java

protected void service(HttpServletRequest req, HttpServletResponse resp)
    throws ServletException, IOException
{
    String method = req.getMethod();

    if (method.equals(METHOD_GET)) {
        ...
        doGet(req, resp);
        ...

    } else if (method.equals(METHOD_HEAD)) {
        ...
        doHead(req, resp);

    } else if (method.equals(METHOD_POST)) {
        doPost(req, resp);
        
    } else if (method.equals(METHOD_PUT)) {
        doPut(req, resp);
        
    } else if (method.equals(METHOD_DELETE)) {
        doDelete(req, resp);
        
    } else if (method.equals(METHOD_OPTIONS)) {
        doOptions(req,resp);
        
    } else if (method.equals(METHOD_TRACE)) {
        doTrace(req,resp);
        
    } else {
        ...
    }
}

簡單起見冻记,用省略號(hào)代替了一些代碼,可以看到来惧,service接到請(qǐng)求后冗栗,會(huì)先判斷請(qǐng)求方法的類型,如果是GET請(qǐng)求就交給doGet去實(shí)現(xiàn)供搀,如果是POST請(qǐng)求就交給doPost去實(shí)現(xiàn)隅居。

對(duì)于一個(gè)具體的servlet(ConcreteHttpServlet)來說,只需要繼承HttpServlet并重寫具體的方法就可以了葛虐。比如某個(gè)Servlet是用來處理GET請(qǐng)求的胎源,那么只重寫doGet方法填寫處理邏輯就OK了。其它的處理不用care屿脐。

不過現(xiàn)在由于Spring等Web框架的出現(xiàn)涕蚤,我們不需要一個(gè)一個(gè)servlet去寫JavaWeb程序,而是由比如Spring的DispatcherServlet這樣的統(tǒng)一的Servlet入口來處理了的诵,它直接映射了”/"這樣的請(qǐng)求URL万栅,從而托管了所有的請(qǐng)求。但是既然是JavaEE框架西疤,還是要遵循Servlet規(guī)范的烦粒,DispatcherServlet的父類FrameworkServlet就對(duì)doGetdoPost等模板方法進(jìn)行了重寫代赁,以滿足規(guī)范的要求扰她。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市管跺,隨后出現(xiàn)的幾起案子义黎,更是在濱河造成了極大的恐慌,老刑警劉巖豁跑,帶你破解...
    沈念sama閱讀 218,036評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件廉涕,死亡現(xiàn)場離奇詭異泻云,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)狐蜕,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,046評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門宠纯,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人层释,你說我怎么就攤上這事婆瓜。” “怎么了贡羔?”我有些...
    開封第一講書人閱讀 164,411評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵廉白,是天一觀的道長。 經(jīng)常有香客問我乖寒,道長猴蹂,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,622評(píng)論 1 293
  • 正文 為了忘掉前任楣嘁,我火速辦了婚禮磅轻,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘逐虚。我一直安慰自己聋溜,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,661評(píng)論 6 392
  • 文/花漫 我一把揭開白布叭爱。 她就那樣靜靜地躺著撮躁,像睡著了一般。 火紅的嫁衣襯著肌膚如雪涤伐。 梳的紋絲不亂的頭發(fā)上馒胆,一...
    開封第一講書人閱讀 51,521評(píng)論 1 304
  • 那天,我揣著相機(jī)與錄音凝果,去河邊找鬼祝迂。 笑死,一個(gè)胖子當(dāng)著我的面吹牛器净,可吹牛的內(nèi)容都是我干的型雳。 我是一名探鬼主播,決...
    沈念sama閱讀 40,288評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼山害,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼纠俭!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起浪慌,我...
    開封第一講書人閱讀 39,200評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤冤荆,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后权纤,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體钓简,經(jīng)...
    沈念sama閱讀 45,644評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡乌妒,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,837評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了外邓。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片撤蚊。...
    茶點(diǎn)故事閱讀 39,953評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖损话,靈堂內(nèi)的尸體忽然破棺而出侦啸,到底是詐尸還是另有隱情,我是刑警寧澤丧枪,帶...
    沈念sama閱讀 35,673評(píng)論 5 346
  • 正文 年R本政府宣布光涂,位于F島的核電站,受9級(jí)特大地震影響豪诲,放射性物質(zhì)發(fā)生泄漏顶捷。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,281評(píng)論 3 329
  • 文/蒙蒙 一屎篱、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧葵蒂,春花似錦交播、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,889評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至永高,卻和暖如春隧土,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背命爬。 一陣腳步聲響...
    開封第一講書人閱讀 33,011評(píng)論 1 269
  • 我被黑心中介騙來泰國打工曹傀, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人饲宛。 一個(gè)月前我還...
    沈念sama閱讀 48,119評(píng)論 3 370
  • 正文 我出身青樓皆愉,卻偏偏與公主長得像,于是被迫代替她去往敵國和親艇抠。 傳聞我的和親對(duì)象是個(gè)殘疾皇子幕庐,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,901評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容

  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語法,類相關(guān)的語法家淤,內(nèi)部類的語法异剥,繼承相關(guān)的語法,異常的語法絮重,線程的語...
    子非魚_t_閱讀 31,631評(píng)論 18 399
  • 一. Java基礎(chǔ)部分.................................................
    wy_sure閱讀 3,811評(píng)論 0 11
  • 定義 模板方法模式是類的行為模式冤寿。準(zhǔn)備一個(gè)抽象類错妖,將部分邏輯以具體方法以及具體構(gòu)造函數(shù)的形式實(shí)現(xiàn),然后聲明一些抽象...
    步積閱讀 883評(píng)論 0 0
  • 這部分主要是與Java Web和Web Service相關(guān)的面試題疚沐。 96暂氯、闡述Servlet和CGI的區(qū)別? 答...
    雜貨鋪老板閱讀 1,405評(píng)論 0 10
  • 晚間打卡 2016/09/13 14/70 今天開始讀《不安的時(shí)候,坐下來寫》1-50亮蛔,完成了任務(wù)痴施。這來書作者緩緩...
    婉琳閱讀 180評(píng)論 0 0