设计模式之模板方法模式

模式背景

有时候,我们可能会遇到这么一种情况,不同的场景下,执行的业务算法的流程或者是骨架是一样的,但是在这些算法中,有一些是可以复用的,而且扩展起来也不是很方便。

二 模板方法模式

2.1 什么是模板方法模式

模板方法模式:固定算法骨架(流程),将一些步骤延迟到子类实现。这样既方便扩展,而且我们可以将一些公共的算法、提取出来。

2.2 优缺点

优点:

1 可以封装不变部分,扩展可变部分

2 提取出公共部分代码,便于维护,也避免代码重复

缺点:

算法骨架如果要修改,则子类都会修改,所以需要尽量是以后不会变的固定算法

2.3 使用场景

1 重构的时候,模板方法是常见的手段

2 多个类有着大致相同的逻辑时,只是有些具体实现不一样,可以使用

四 代码示例

public abstract class AbstractTemplate {

    public abstract boolean validateParams(Integer itemId);

    public  <T> T queryItem(Class clazz, Integer itemId) {
        return (T)Repository.queryItem(clazz,itemId);
    }

    public abstract <T> String buildSeoLink(T item);

    public <T> String buildSeoURL(Class<T> clazz, Integer itemId) {
        boolean valid = validateParams(itemId);
        if (!valid) {
            throw new IllegalArgumentException("[buildSeoURL] 提供的参数非法,请检查");
        }

        T item = queryItem(clazz, itemId);
        if (item == null) {
            return null;
        }
        return buildSeoLink(item);
    }
}



public class CategorySeoURLTemplate extends AbstractTemplate {



    public boolean validateParams(Integer itemId) {

        return itemId < 1000;

    }



    public <T> String buildSeoLink(T item) {

        if (item == null) {

            return null;

        }

        Category category = (Category)item;

        StringBuilder sb = new StringBuilder();

        if (category.getParent() != null) {

            String parentSeoLink = buildSeoLink(category.getParent());

            sb.append(parentSeoLink);

        }

        String displayName = category.getDisplayName();

        String seoLink = StringUtils.formatSeoURL(displayName,' ','-',Boolean.TRUE);

        sb.append("/").append(seoLink);

        return sb.toString();

    }

}


public class ProductSeoURLTemplate extends AbstractTemplate {

    public boolean validateParams(Integer itemId) {

        return itemId > 1000 || itemId < 10000;

    }



    public <T> String buildSeoLink(T item) {

        if (item == null) {

            return null;

        }

        Product product = (Product)item;

        StringBuilder sb = new StringBuilder();

        String displayName = product.getDisplayName();

        String seoLink = StringUtils.formatSeoURL(displayName,' ','-',Boolean.TRUE);

        sb

            .append("/")

            .append(seoLink)

            .append("?pid=")

            .append(product.getProductId());



        return sb.toString();

    }



}


public class SKUSeoURLTemplate extends AbstractTemplate {

    public boolean validateParams(Integer itemId) {

        return itemId > 100000 || itemId < 100000;

    }





    public <T> String buildSeoLink(T item) {

        if (item == null) {

            return null;

        }

        SKU sku = (SKU)item;

        StringBuilder sb = new StringBuilder();

        String seoName = sku.getSeoName();

        String seoLink = StringUtils.formatSeoURL(seoName,' ','-',Boolean.TRUE);

        sb.append("/").append(seoLink);

        sb.append("-")

            .append(sku.getProduct()

            .getProductId())

            .append("?sid=")

            .append(sku.getSkuId());



        return sb.toString();

    }

}


public class Repository<T> {

    private static Map<Integer,Category> categoryDB = new HashMap<Integer, Category>();

    private static Map<Integer,Product> productDB = new HashMap<Integer, Product>();

    private static Map<Integer,SKU> skuDB = new HashMap<Integer, SKU>();





    static {

        Category c1 = new Category(1,"Household Electrical Appliances",null);

        Category c12 = new Category(12,"Television",c1);

        Category c121 = new Category(121,"Smart Television",c12);

        Category c2 = new Category(2,"Office",null);

        Category c21 = new Category(21,"Computer",c2);

        Category c211 = new Category(211,"Laptop",c21);

        categoryDB.put(c1.getCategoryId(),c1);

        categoryDB.put(c12.getCategoryId(),c12);

        categoryDB.put(c121.getCategoryId(),c121);

        categoryDB.put(c2.getCategoryId(),c2);

        categoryDB.put(c21.getCategoryId(),c21);

        categoryDB.put(c211.getCategoryId(),c211);

        Product p1 = new Product(1000,"Sunglasses");

        Product p2 = new Product(2000,"Premium Aviator Sunglasses");

        productDB.put(p1.getProductId(),p1);

        productDB.put(p2.getProductId(),p1);

        SKU s1 = new SKU(100001,"Red Sunglasses","sunglassess red",p1);

        SKU s2 = new SKU(100002,"Black Sunglasses","sunglassess black",p1);

        SKU s3 = new SKU(200010,"Men Premium Aviator Sunglasses","men premium aviator sunglasses",p2);

        SKU s4 = new SKU(200012,"Women Premium Aviator Sunglasses","women premium aviator sunglasses black",p2);

        skuDB.put(s1.getSkuId(),s1);

        skuDB.put(s2.getSkuId(),s2);

        skuDB.put(s3.getSkuId(),s3);

        skuDB.put(s4.getSkuId(),s4);

    }



    public static Object queryItem(Class clazz, Integer itemId) {

        if (clazz == Category.class) {

            return categoryDB.get(itemId);

        } else if (clazz == Product.class) {

            return productDB.get(itemId);

        } else if (clazz == SKU.class) {

            return skuDB.get(itemId);

        }

        return null;

    }

}


public  class StringUtils {

    public static boolean isBlank(String str) {

        return str == null || str.trim().equals("");

    }



    public static boolean isNotBlank(String str) {

        return str != null && !str.trim().equals("");

    }



    public static boolean isAlphanumericSpace(char ch) {

        return Character.isLetterOrDigit(ch) || (ch == ' ');

    }

    public static final String formatSeoURL(String pSrcLink, char pSrcDelim, char pDestDelim, boolean toLowerCase){

        if (isBlank(pSrcLink)) {

            return null;

        }



        if (pSrcLink.indexOf("&") != -1) {

            pSrcLink = pSrcLink.replaceAll("&", "and");

        }



        char[] chars = pSrcLink.trim().toCharArray();

        for(int i = 0; i < chars.length; i++){

            if(chars[i] == '\'' || chars[i] == '-'){

                continue;

            }

            if(isAlphanumericSpace(chars[i])){

                continue;

            }

            pSrcLink = remove(pSrcLink, chars[i]);

        }

        if(toLowerCase){

            pSrcLink = pSrcLink.toLowerCase();

        }

        pSrcLink = pSrcLink.replaceAll("\\s{1,}", " ");

        pSrcLink = pSrcLink.replace(pSrcDelim, pDestDelim);

        return pSrcLink;

    }



    public static String remove(String str, char remove) {

        if ((isBlank(str)) || (str.indexOf(remove) == -1)) {

            return str;

        }

        char[] chars = str.toCharArray();

        int pos = 0;

        for (int i = 0; i < chars.length; i++) {

            if (chars[i] != remove) {

                chars[(pos++)] = chars[i];

            }

        }

        return new String(chars, 0, pos);

    }

}


public class Client {

    public static void main(String[] args) {

        AbstractTemplate template = new CategorySeoURLTemplate();

        String  catSeoURL = template.buildSeoURL(Category.class, 121);

        System.out.println(catSeoURL);





        // 切换模板

        template = new ProductSeoURLTemplate();

        String  prodSeoURL = template.buildSeoURL(Product.class, 2000);

        System.out.println(prodSeoURL);



        template = new SKUSeoURLTemplate();

        String  skuSeoURL = template.buildSeoURL(SKU.class, 200012);

        System.out.println(skuSeoURL);

    }

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
模板方法设计模式是一种行为设计模式,它定义了一个算法的骨架,将一些步骤的实现延迟到子类中。这种模式可以确保算法的结构保持不变,但允许子类提供特定的实现细节。在C++中,模板方法设计模式可以通过使用虚函数和继承来实现。 在给出的引用中,我们可以看到一个名为CoffeineBeverage的类,它定义了一个prepare_recipe()方法,并在内部调用了其他几个私有方法。这个类是一个基类,可以被子类继承并重写其中的方法。 具体来说,模板方法设计模式的关键是将算法的骨架定义在基类中,而将可变的实现细节留给子类去实现。在这个例子中,prepare_recipe()方法是算法的骨架,而brew()和add_condiments()方法则是可变的实现细节。 通过将brew()和add_condiments()方法定义为虚函数,基类CoffeineBeverage允许子类去重写这些方法以提供自己的实现。这样,当调用prepare_recipe()方法时,实际执行的是子类中重写后的方法。 使用模板方法设计模式的好处是可以提高代码的复用性和可扩展性。算法的骨架在基类中只需要定义一次,而具体的实现细节可以在不同的子类中灵活变化。 总结起来,C++中的模板方法设计模式通过定义一个算法的骨架并将可变的实现细节留给子类去实现,可以提供代码的复用性和可扩展性。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [C++设计模式模板方法模式](https://blog.csdn.net/Long_xu/article/details/127118813)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* [C++设计模式-模板方法模式](https://blog.csdn.net/qq78442761/article/details/91990149)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

莫言静好、

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值