国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

聊一聊模板方法模式

開發(fā) 前端
模板方法模式其實(shí)是一個(gè)比較簡(jiǎn)單的設(shè)計(jì)模式,它有如下優(yōu)點(diǎn):1、封裝不變的邏輯,擴(kuò)展差異化的邏輯;2、抽取公共代碼,提高代碼的復(fù)用性;3、父類控制行為,子類實(shí)現(xiàn)細(xì)節(jié)。

一、概述

模板方法模式,又叫模板模式,屬于23種設(shè)計(jì)模式中的行為型模式。在抽象類中公開定義了執(zhí)行的方法,子類可以按需重寫其方法,但是要以抽象類中定義的方式調(diào)用方法。總結(jié)起來就是:定義一個(gè)操作的算法結(jié)構(gòu),而將一些步驟延遲到子類中。在不改變算法結(jié)構(gòu)的情況下,子類能重定義該算法的特定步驟。

下面是模板模式的UML圖,抽象類(AbstractClass)定義公共的步驟和方法,依次調(diào)用實(shí)際的模板方法,當(dāng)然每個(gè)方法可以是抽象方法(需交給子類實(shí)現(xiàn)),也可以是提供默認(rèn)的方法。具體的類(ConcreteClass)可以重寫所有的方法,但是不能改變抽象類中定義的整體結(jié)構(gòu)。

圖片

二、入門案例

相信大家都吃過蛋糕,現(xiàn)在市面上的蛋糕可謂是五花八門,你能想到的造型商家能給你整出來,你想不到的,他們也能整出來。不過無論造型如何變化,不變的有兩種東西:“奶油”和“面包”。其余的材料隨意搭配,就湊成了各式各樣的蛋糕。

圖片

基于這個(gè)場(chǎng)景,我們來寫一個(gè)案例,進(jìn)一步了解下模板模式;創(chuàng)建三個(gè)類:Cake(蛋糕)、StrawberryCake(草莓蛋糕)、CherryCake(櫻桃蛋糕)。最后創(chuàng)建一個(gè)Client類,實(shí)現(xiàn)這個(gè)制作蛋糕的調(diào)用過程。

package com.wsrf.template;

/**
 * @author 往事如風(fēng)
 * @version 1.0
 * @date 2023/5/4 16:12
 * @description:抽象類:蛋糕
 */
public abstract class Cake {

    /**
     * 制作
     */
    public void make() {
        System.out.println("開始準(zhǔn)備材料。");
        bread();
        cream();
        fruit();
        System.out.println("經(jīng)過一系列的操作。");
        System.out.println("制作完成。");
    }

    /**
     * 準(zhǔn)備面包
     */
    public void bread() {
        System.out.println("準(zhǔn)備材料:面包");
    }

    /**
     * 準(zhǔn)備奶油
     */
    public void cream() {
        System.out.println("準(zhǔn)備材料:奶油");
    }

    /**
     * 準(zhǔn)備水果
     */
    protected abstract void fruit();

}
package com.wsrf.template;

/**
 * @author 往事如風(fēng)
 * @version 1.0
 * @date 2023/5/4 16:13
 * @description:具體類:草莓蛋糕
 */
public class StrawberryCake extends Cake{
    @Override
    protected void fruit() {
        System.out.println("準(zhǔn)備材料:草莓");
    }
}
package com.wsrf.template;

/**
 * @author 往事如風(fēng)
 * @version 1.0
 * @date 2023/5/4 16:14
 * @description:具體類:櫻桃蛋糕
 */
public class CherryCake extends Cake{
    @Override
    protected void fruit() {
        System.out.println("準(zhǔn)備材料:櫻桃");
    }
}
package com.wsrf.template;

/**
 * @author 往事如風(fēng)
 * @version 1.0
 * @date 2023/5/4 16:21
 * @description
 */
public class Client {

    public static void main(String[] args) {
        Cake c1 = new CherryCake();
        c1.make();
        System.out.println("-------------------------------------");
        Cake c2 = new StrawberryCake();
        c2.make();
    }
}
/**
輸出結(jié)果:
開始準(zhǔn)備材料。
準(zhǔn)備材料:面包
準(zhǔn)備材料:奶油
準(zhǔn)備材料:櫻桃
經(jīng)過一系列的操作。
制作完成。
-------------------------------------
開始準(zhǔn)備材料。
準(zhǔn)備材料:面包
準(zhǔn)備材料:奶油
準(zhǔn)備材料:草莓
經(jīng)過一系列的操作。
制作完成。
*/

在Cake類中定義了制作蛋糕的整個(gè)步驟,也就是make方法;然后抽取了公用的方法,bread方法和cream方法;最后定義一個(gè)抽象方法fruit,這個(gè)方法需要交給具體的子類StrawberryCake和CherryCake去實(shí)現(xiàn),從而定制差異化的“蛋糕”。

三、運(yùn)用場(chǎng)景

通過上面的“蛋糕”案例,在平時(shí)開發(fā)中我們可以具體分析一下業(yè)務(wù)需求,首先在父類中定義需求需要實(shí)現(xiàn)的步驟,然后將可以公用的方法抽取到父類中,將個(gè)性化的方法放到具體的子類中去實(shí)現(xiàn);這樣可以很好的培養(yǎng)“抽象化”的思維模式,這是拉開差距的第一步。

最近在開發(fā)中,遇到這樣的一個(gè)業(yè)務(wù)場(chǎng)景:需要給不同的管理人員計(jì)算各種不同的津貼,如區(qū)域總監(jiān)有區(qū)域管理津貼、傭金、培養(yǎng)育成津貼等等。通過分析,每種不同類型的津貼,都是需要金額x比例x系數(shù),比如每種津貼都有不同的計(jì)算方式,系數(shù)也是。所以,大致的想法就是:金額x比例x系數(shù)這個(gè)計(jì)算方式設(shè)置為統(tǒng)一的方法,系數(shù)和比例讓具體的津貼子類去實(shí)現(xiàn)。所以大致的偽代碼如下;

首先,我定義了一個(gè)抽象類AbstractManageAllowanceCalService,用于定義統(tǒng)一的計(jì)算方法,并預(yù)留了獲取比例和獲取系數(shù)的抽象方法。

/**
 * @author 往事如風(fēng)
 * @version 1.0
 * @date 2023/5/4 17:12
 * @description:津貼計(jì)算父類
 */
@Slf4j
public abstract class AbstractManageAllowanceCalService {

    /**
     * 計(jì)算津貼
     * @param amount
     * @return
     */
    public BigDecimal calAmount(BigDecimal amount) {
        if (Objects.isNull(amount)) {
            return BigDecimal.ZERO;
        }
        BigDecimal ratio = getRatio();
        BigDecimal coefficient = getCoefficient();
        log.info("金額:{},系數(shù):{},比例:{}", amount, coefficient, ratio);
        return amount.multiply(ratio).multiply(coefficient);
    }

    /**
     * 獲取比例
     * @return
     */
    protected abstract BigDecimal getRatio();

    /**
     * 獲取系數(shù)
     * @return
     */
    protected abstract BigDecimal getCoefficient();
}

然后,定義兩個(gè)具體的子類,用于計(jì)算區(qū)域管理津貼和傭金。

/**
 * @author 往事如風(fēng)
 * @version 1.0
 * @date 2023/5/4 17:17
 * @description:區(qū)域管理津貼計(jì)算
 */
@Service
public class AreaBusinessAllowanceCalService extends AbstractManageAllowanceCalService{
    /**
     * 區(qū)域管理津貼比例
     * @return
     */
    @Override
    protected BigDecimal getRatio() {
        return new BigDecimal(0.5).setScale(1, BigDecimal.ROUND_HALF_UP);
    }

    /**
     * 區(qū)域管理津貼系數(shù)
     * @return
     */
    @Override
    protected BigDecimal getCoefficient() {
        return new BigDecimal(0.92).setScale(2, BigDecimal.ROUND_HALF_UP);
    }
}
/**
 * @author 往事如風(fēng)
 * @version 1.0
 * @date 2023/5/4 17:19
 * @description:傭金計(jì)算
 */
@Service
public class SalaryCalService extends AbstractManageAllowanceCalService{
    /**
     * 傭金比例
     * @return
     */
    @Override
    protected BigDecimal getRatio() {
        return new BigDecimal(0.45).setScale(2, BigDecimal.ROUND_HALF_UP);
    }

    /**
     * 傭金系數(shù)
     * @return
     */
    @Override
    protected BigDecimal getCoefficient() {
        return new BigDecimal(0.88).setScale(2, BigDecimal.ROUND_HALF_UP);
    }
}

最后,定義一個(gè)controller類,用于接口調(diào)用,提供計(jì)算能力;接收兩個(gè)參數(shù),金額和計(jì)算津貼類型。

/**
 * @author 往事如風(fēng)
 * @version 1.0
 * @date 2023/5/4 17:21
 * @description
 */
@RestController
@RequestMapping("/cal")
public class CalController implements ApplicationContextAware {

    private static ApplicationContext applicationContext;

    @PostMapping("/amount")
    public Result<BigDecimal> calAmount(BigDecimal amount, String calType) {
        AbstractManageAllowanceCalService service = null;
        if ("AREA".equals(calType)) {
            // 區(qū)域管理津貼
             service = (AbstractManageAllowanceCalService) applicationContext.getBean("areaBusinessAllowanceCalService");
        } else if ("SALARY".equals(calType)) {
            // 傭金
            service = (AbstractManageAllowanceCalService) applicationContext.getBean("salaryCalService");
        }
        if (Objects.nonNull(service)) {
            return Result.success(service.calAmount(amount));
        }
        return Result.fail();
    }

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        CalController.applicationContext = applicationContext;
    }
}

在這個(gè)controller類中,我通過分析“類型”這個(gè)參數(shù),來判斷需要調(diào)用哪個(gè)service去實(shí)現(xiàn)具體的計(jì)算邏輯。這里用了if-else的方式去實(shí)現(xiàn);其實(shí)也可以用到另一個(gè)設(shè)計(jì)模式——策略模式,這樣寫出來的代碼就會(huì)比較優(yōu)雅,這里就不對(duì)策略模式展開贅述了。

四、源碼中運(yùn)用

4.1、JDK源碼中的模板模式

在JDK中其實(shí)也有很多地方運(yùn)用到了模板模式,這里咱挑一個(gè)講。并發(fā)包下的AbstractQueuedSynchronizer類,就是一個(gè)抽象類,也就是我們先前的文章中提到過的AQS。

public abstract class AbstractQueuedSynchronizer
    extends AbstractOwnableSynchronizer
    implements java.io.Serializable {
 
    public final void acquire(int arg) {
        if (!tryAcquire(arg) &&
            acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
            selfInterrupt();
    }

 public final boolean release(int arg) {
        if (tryRelease(arg)) {
            Node h = head;
            if (h != null && h.waitStatus != 0)
                unparkSuccessor(h);
            return true;
        }
        return false;
    }

    protected boolean tryAcquire(int arg) {
        throw new UnsupportedOperationException();
    }

    protected boolean tryRelease(int arg) {
        throw new UnsupportedOperationException();
    }
}

其中,tryAcquire和tryRelease這兩個(gè)方式直接拋了異常,用protected關(guān)鍵詞修飾,需要由子類去實(shí)現(xiàn)。然后在acquire和release方法中分別去調(diào)用這兩方法。也就是acquire方法定義了一個(gè)統(tǒng)一的結(jié)構(gòu),差異化的tryAcquire方法需要具體的子類去實(shí)現(xiàn)功能,實(shí)現(xiàn)了模板模式。

4.2、Spring源碼中的模板模式

說到源碼,Spring是一個(gè)繞不開的話題,那就來學(xué)習(xí)下Spring中的模板模式。其中,有一個(gè)類DefaultBeanDefinitionDocumentReader,它是BeanDefinitionDocumentReader的實(shí)現(xiàn)類,是提取spring配置文件中的bean信息,并轉(zhuǎn)化為BeanDefinition。

public class DefaultBeanDefinitionDocumentReader implements BeanDefinitionDocumentReader {
 
    protected void doRegisterBeanDefinitions(Element root) {
        BeanDefinitionParserDelegate parent = this.delegate;
        this.delegate = this.createDelegate(this.getReaderContext(), root, parent);
        //...

        this.preProcessXml(root);
        this.parseBeanDefinitions(root, this.delegate);
        this.postProcessXml(root);
        this.delegate = parent;
    }

    protected void preProcessXml(Element root) {
    }

    protected void postProcessXml(Element root) {
    }
}

這里我截圖了其中的一段代碼,主要是doRegisterBeanDefinitions這個(gè)方法,從跟節(jié)點(diǎn)root出發(fā),root下的每個(gè)bean注冊(cè)定義。

該方法中還調(diào)用了preProcessXml和postProcessXml這兩個(gè)方法,但是在DefaultBeanDefinitionDocumentReader類中,這兩個(gè)方法是未實(shí)現(xiàn)的,需要其子類去實(shí)現(xiàn)具體的邏輯。所以,這里也是一個(gè)很典型的模板模式的運(yùn)用。

五、總結(jié)

模板方法模式其實(shí)是一個(gè)比較簡(jiǎn)單的設(shè)計(jì)模式,它有如下優(yōu)點(diǎn):1、封裝不變的邏輯,擴(kuò)展差異化的邏輯;2、抽取公共代碼,提高代碼的復(fù)用性;3、父類控制行為,子類實(shí)現(xiàn)細(xì)節(jié)。

其缺點(diǎn)就是不同的實(shí)現(xiàn)都需要一個(gè)子類去維護(hù),會(huì)導(dǎo)致子類的個(gè)數(shù)不斷增加,造成系統(tǒng)更加龐大。

用一句話總結(jié):將公用的方法抽取到父類,在父類中預(yù)留可變的方法,最后子類去實(shí)現(xiàn)可變的方法。

模板模式更多的是考察我們對(duì)于公用方法的提取;對(duì)于編程也是這樣,更多的是一種思維能力,不能只局限于代碼,要把格局打開。

六、參考源碼

編程文檔:
https://gitee.com/cicadasmile/butte-java-note

應(yīng)用倉庫:
https://gitee.com/cicadasmile/butte-flyer-parent
責(zé)任編輯:武曉燕 來源: 知了一笑
相關(guān)推薦

2022-11-26 00:00:06

裝飾者模式Component

2022-11-01 08:46:20

責(zé)任鏈模式對(duì)象

2023-07-25 15:06:39

2022-06-01 09:51:51

Golang方法接收者

2022-09-26 08:03:25

VMware虛擬機(jī)

2023-02-09 10:39:15

gRPC通信模式

2023-12-14 11:35:32

.NET泄露模式

2018-06-07 13:17:12

契約測(cè)試單元測(cè)試API測(cè)試

2023-09-22 17:36:37

2020-05-22 08:16:07

PONGPONXG-PON

2021-01-28 22:31:33

分組密碼算法

2021-05-12 18:02:23

方法創(chuàng)建線程

2023-03-10 07:47:41

克隆jQuery

2020-08-12 08:34:16

開發(fā)安全We

2022-10-08 11:33:56

邊緣計(jì)算云計(jì)算

2020-06-28 09:30:37

Linux內(nèi)存操作系統(tǒng)

2021-01-01 09:01:05

前端組件化設(shè)計(jì)

2020-09-08 06:54:29

Java Gradle語言

2019-12-17 10:06:18

CDMA高通4G

2022-03-29 09:56:21

游戲版本運(yùn)營(yíng)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

91精品在线免费| 黄色高清在线观看| 97色伦图片97色伦在线电影| 欧美性生活一级片| 亚洲成人免费在线| 欧洲高清一区二区| 成人三级黄色免费网站| 奇米综合一区二区三区精品视频| 69堂国产成人免费视频| 国产3p露脸普通话对白| 国产一区99| 日韩国产精品一区| 欧美大学生性色视频| 午夜亚洲成人| 成人精品鲁一区一区二区| 国产精品一区av| 亚洲高清影院| 欧美日韩激情在线| 日本美女高潮视频| 久久精品国产亚洲a| 亚洲2020天天堂在线观看| 成人亚洲综合天堂| 中文字幕亚洲视频| 神马影院一区二区| 91一区在线| 久久久久久久久久久人体| 日日躁夜夜躁人人揉av五月天| 日韩精品久久理论片| 国产精品视频免费在线观看| 在线视频超级| 欧美一卡2卡3卡4卡| 中文字幕一二三区在线观看| 国产午夜精品久久久久久免费视 | 9999精品| 亚洲精品一区二区久| 国产在线传媒| 亚洲成人手机在线| 国产传媒视频在线观看| 亚洲视频在线一区二区| 人人妻人人添人人爽欧美一区| 久久精品综合| 日本在线观看一区二区| 久久精品官网| 日韩激情视频| 美国十次了思思久久精品导航| 国产成人精品综合久久久| 欧美系列电影免费观看| 国产+成+人+亚洲欧洲| 国产麻豆精品| 欧美国产第二页| 国产精品美女久久久久久不卡 | 成人在线观看毛片| 青青草国产精品97视觉盛宴| www污在线观看| 久久久久免费观看| 日本wwww视频| 国产日韩欧美在线一区| 调教在线观看| 疯狂蹂躏欧美一区二区精品| 岛国视频免费在线观看| 日韩色视频在线观看| 丰满大乳少妇在线观看网站| 在线亚洲国产精品网| 国产日产精品_国产精品毛片| 97视频资源在线观看| 欧美日韩精品| 在线一级视频| 91久久久久久久久久久久久| 一区二区三区在线视频免费| 九九久久婷婷| 尤物视频网站在线观看| 亚洲综合视频1区| 五月天欧美精品| 成人片免费看| 日韩在线中文字幕| 一个人免费视频www在线观看| 国内精品美女在线观看| 国产99久久精品一区二区| 日韩中文字幕一区二区三区| 亚洲在线第一页| 蜜桃精品视频| 在线观看亚洲精品| 91传媒免费视频| 激情五月色综合国产精品| 欧美日韩国产精品专区| 日韩一级性生活片| av日韩在线播放| 精品国精品自拍自在线| 亚洲韩国在线| 色999国产精品| 欧美日精品一区视频| 中文精品无码中文字幕无码专区| 亚洲成人av观看| 日韩毛片在线免费观看| 男人的天堂成人| 伊人激情综合| 久久久久久久香蕉网| 97精品国产99久久久久久免费| 中文字幕一区免费在线观看| 日韩精品无码一区二区三区免费| 极品日韩av| 农村寡妇一区二区三区| 亚洲成色999久久网站| av电影天堂一区二区在线| 理论片一区二区在线| 麻豆app在线观看| 亚洲天堂第一区| 国产精品日韩精品| 国产精品1区2区3区| 午夜欧美激情| 中文字幕日本最新乱码视频| 久久精品99国产精品酒店日本| 久久se精品一区二区| 97超碰免费在线| 亚洲电影免费| 亚洲人午夜色婷婷| 国产拍揄自揄精品视频麻豆 | 精品毛片免费观看| 奇米影视四色在线| 国内精品久久久久伊人av | 久久视频一区| 欧美24videosex性欧美| 天天想你在线观看完整版电影免费 | 国产农村妇女毛片精品久久麻豆 | 欧美一区=区三区| wwwwww.色| 成人av中文| 欧美精品在线视频观看| 亚洲精品大片www| 国产模特精品视频久久久久| 亚洲第一会所001| 中文字幕第5页| 亚洲视频欧美在线| 国产福利成人在线| 亚洲精品久久久久国产| 亚洲视频一区在线| 精品一区二区三区av| 香蕉久久精品| 美足av综合网| 手机看片1024久久| 男人日女人逼逼| 久草一区二区| 国产精品成人av在线| 亚洲美女视频网站| 高潮白浆女日韩av免费看| 久久毛片高清国产| 男人操女人的视频在线观看欧美| 国产精品免费大片| 在线成人免费| 乡村艳史在线观看| 欧美激情二区| 九色视频在线播放| 丁香激情视频| 人妻无码久久一区二区三区免费| 久久精品99| 粉嫩av四季av绯色av第一区| 欧美一级视频免费在线观看| 欧美国产日韩视频| 中文欧美日本在线资源| 亚洲欧美国产视频| 亚洲欧美日韩国产中文专区| 日韩精品极品视频免费观看| 欧美va亚洲va在线观看蝴蝶网| 精品久久久久香蕉网| 精品国产91久久久久久久妲己| 欧美精品xxxxbbbb| 欧美一区二区三区影视| 欧美在线你懂的| 色综合视频在线观看| 欧美日韩激情小视频| 大伊人狠狠躁夜夜躁av一区| 欧美在线你懂的| 欧美主播一区二区三区美女| 欧美日韩国产一区中文午夜| 色综合一区二区| 日韩欧美综合一区| 日韩大片免费观看视频播放| 亚洲欧美日韩直播| 久久精品国产2020观看福利| 久久久久久国产| 91九色视频在线| 久久精品一二三区| 男人j进女人j| 91福利国产在线观看菠萝蜜| 欧美一区在线看| 欧美日韩在线视频观看| 国产成人在线播放| 中文字幕 91| 欧美美女在线直播| 国产日韩欧美一区二区三区乱码| 中文字幕亚洲国产| 加勒比成人在线| 国产欧美日韩电影| 成人性生交大片免费看中文| 亚洲精品视频免费| 香蕉久久夜色| 日本色护士高潮视频在线观看 | 亚洲精品色图| 国产 日韩 欧美| 99在线播放|