• 用最简单的例子理解装饰器模式(Decorator Pattern)


    假设有一个公司要做产品套餐,即把不同的产品组合在一起,不同的组合对应不同的价格。最终呈现出来的效果是:把产品组合的所有元素呈现出来,并显示该组合的价格。

     

    每个产品都有名称和价格,首先设计一个关于产品的抽象基类。

        public abstract class ProductBase
    
        {
    
            public abstract string GetName();
    
            public abstract double GetPrice();
    
        }

     

    所有的产品都必须继承这个基类,比如家居用品、电器产品等,把这些具体的产品提炼成一个继承ProductBase的子类。

     

        public class ConcretProuct : ProductBase
    
        {
    
            private string _name;
    
            private double _price;
    
            public ConcretProuct(string name, double price)
    
            {
    
                this._name = name;
    
                this._price = price;
    
            }
    
            public override string GetName()
    
            {
    
                return _name;
    
            }
    
            public override double GetPrice()
    
            {
    
                return _price;
    
            }
    
        }
    

    然后考虑产品组合。比如卖平底锅,可能送酱油,也有可能送酱油+老坛酸菜,可能的组合包括:
    ○ 平底锅
    ○ 平底锅 + 酱油
    ○ 平底锅 + 酱油 + 老坛酸菜   

     

    在这里,可以把酱油,老坛酸菜看作是装饰器,因为每加一个产品,都是在原有的基础上增加的。比如做"平底锅 + 酱油"这个组合,是在"平底锅"的基础上增加了"酱油"。

     

    现在把酱油、老坛酸菜也设计成继承ProductBase的子类,也就是装饰器类。不过,与ConcretProuct类不同的是,装饰器类需要引用ProductBase,在这里,无论是显示产品组合还是计算产品产品组合价格,都离不开这个引用的ProductBase。

     

       public class Decorator : ProductBase
    
        {
    
            private ProductBase _product = null;
    
            private string _name;
    
            private double _price;
    
            public Decorator(ProductBase product, string name, double price)
    
            {
    
                this._product = product;
    
                this._name = name;
    
                this._price = price;
    
            }
    
            public override string GetName()
    
            {
    
                return string.Format("{0},{1}", _product.GetName(), _name);
    
            }
    
            public override double GetPrice()
    
            {
    
                return _product.GetPrice() + _price;
    
            }
    
        }
    

    以上,显示产品名称的时候,把装饰器类Decorator引用的ProductBase的名称和当前名称组合起来,以逗号分隔;显示产品价格的时候,把引用的ProductBase的价格和当前价格相加。

     

    客户端如下:

        class Program
    
        {
    
            static void Main(string[] args)
    
            {
    
                ConcretProuct livingProduct = new ConcretProuct("平底锅",100);
    
                Console.WriteLine(PrintProductDetails(livingProduct));
    
                Decorator dec1 = new Decorator(livingProduct,"海鲜酱油",10);
    
                Console.WriteLine(PrintProductDetails(dec1));
    
                Decorator dec2 = new Decorator(dec1, "老坛酸菜",12);
    
                Console.WriteLine(PrintProductDetails(dec2));
    
                Console.ReadKey();
    
            }
    
            private static string PrintProductDetails(ProductBase product)
    
            {
    
                return string.Format("产品组合:{0}     价格:{1}", product.GetName(), product.GetPrice());
    
            }
    
        } 
    

    1 

      

  • 相关阅读:
    解除mac默认线程限制
    pd windows虚拟机下安装子系统
    国内外最好的12款项目管理系统优劣势分析
    如何建立自己的看板系统
    如何做好产品管理
    为什么都认为confluence非常好,好在哪里
    瀑布式项目管理软件测评!
    看板方法的定义、原则和实践
    类似Confluence的软件有哪些
    CSCMS代码审计
  • 原文地址:https://www.cnblogs.com/darrenji/p/3961072.html
Copyright © 2020-2023  润新知