ThinkChat2.0新版上线,更智能更精彩,支持会话、画图、阅读、搜索等,送10W Token,即刻开启你的AI之旅 广告
**本文来自:崔成龙博客专栏。转载请注明出处:**[**http://blog.csdn.net/xiaoxian8023**](http://blog.csdn.net/xiaoxian8023)    软考上午题终于考完了。三个赶考者都感觉不错。检查了2遍,提前30分钟都出来了 。   小A,小B,小C楼下碰头,相视一笑,轻松之感溢于言表。遂决定去吃面,以犒劳自己的肚子。  “老板,我要西红柿鸡蛋面!”,“尖椒炸酱面!”,“苏格兰打卤面!”。。。。。。“好嘞!”   面快出锅了,“哎哎,老板,怎么我的面跟他的面一样啊,就是换了一下卤?”往面里放卤的服务员翻了小B一眼,不搭理。旁边那个年长点的师傅笑了笑,说道“小兄弟儿,这你就不懂了吧。其实面都是一样的,只是换一下卤罢了。”“哦哦,原来如此啊。”小B灰溜溜得端起面来跑开了。。。   面毕,小B实在是咽不下这口气。就想在小C那儿“捞回点儿本来”。   “小C,面吃的怎么样啊?”小B阴阳怪气的问道。   小C看着小B的表情,觉得有点不对劲,自己是不是又要上当了,但是又想不出来自己哪留有什么“把柄”,然后就答道,   “还,还行吧,问这干嘛??”   “嘿,你怕什么呀。我考一个关于吃面的题目”   “嗨,我还以为什么呢,随便考吧,哥是来者不拒。”小C心里想到,就你那智商,能出什么样的难题呢,等着被鄙视吧。   “你感觉自家煮面和饭馆煮面有什么不同?”   “这还用问!感觉不同呗。自己煮面,自己累死了还得做饭,又是炒卤,又是煮面的,最低也得折腾半小时吧。饭馆多好啊,只要吆喝一声,人家都给你做好,端到你面前了。而且速度超快的。连这个都不知道。也对,就你那智商,好吧,我原谅你,哈哈哈哈”   “哼,别笑得太早了。同样是面?你说为什么感觉不同啊”   “呃,这个嘛,嗯,让我想想”小C装傻了。。。   “哈,有点意思哎。我给你说说看看对不。”小A憋不住了。接着说道,   “刚才小C说了,自家煮面,你必须自己做卤,自己煮面。每次想吃面的时候,这两步都是少不了的。但是面馆不一样啊。饭馆卤都是提前做好的。你点了面后,面馆给你煮面,然后加上你要的卤不就OK了吗!而且速度超快。比自己做饭省事多了。”   “哦~,原来如此。”    “我说那会那个服务员像看白痴的一样看我呀。”小B嘀咕道   “what,what!!原来你被吃瘪了啊,哈哈”小C得意得捧腹大笑。   小A眼睛一转,对即将要发飙的小B说道,   “既然是你提出来的,你说说饭馆为什么要这么做呢?”   “呃,嗯,,,,我想,应该是为了更快捷,方便地方便用户,同时提高自己的效率吧。”   “还有没有别的?”   小B想了半天,摇来摇头,“想不到了”   “提示一点,面馆在增加新类型面的时候,是怎么做的?”   “哦~~,我知道了,因为面差不多是一样的,只需要准备卤就行了。”   “嗯,对的。其实面也是有不同的。比如说宽面,拉面,圆面。这样一组合,新的类型的面就出炉了。”   “原来面馆里也有这么大学问啊。”   “那是当然。365行,行行出状元啊。先不说这个了,既然说道这个面了,那你用面向对象写写刚才我们讨论的这件事儿吧。”   “好。我试试吧”小B自信道。    既然是面向对象,那么先把类找出来吧。首先有原料(Material),下分为面(*Noodles*)和卤(*Halogen*)。面有宽面条(WideNoodles),窄面条(NarrowNoodles)等多种类型,卤菜有西红柿鸡蛋卤(TomatoAndEgg),茄子豆瓣酱(AubergineBeanPaste)等多种卤菜。  西红柿鸡蛋面类(TomatoEggNoodles)为西红柿鸡蛋卤和宽面构成,而苏格兰打卤面(ScotlandNoodles)则是有茄子 豆瓣酱和窄面构成的。 先画出UML图: uml图1.0版 ![](https://box.kancloud.cn/2016-01-14_56970cf226f54.png)    “看看怎么样啊”,小B得瑟的说道。    “你有没有考虑代码的实现呢?”小A用看白痴的眼神扫了小B一眼,“除了C++,还有几个面向对象的有多继承啊!”    “呃,这个,失误失误,我马上改!”    “别急啊,还有错误呢,西红柿鸡蛋面有两部分构成,那肯定是用聚合或者是组合啊,不应该用继承的”小A提醒道。    “哦,原来如此,我说怎么感觉有点不对劲呢,等着吧”    5分钟过后。。。    “哦,终于出来了,看看这次怎么样啊。” uml图2.0.版 ![](https://box.kancloud.cn/2016-01-14_56970cf23a438.png)   “西红柿鸡蛋面和苏格兰打卤面的组成关系也画出来了。嗯不错,不过还有一点不对劲。假如我现在加一个新类型的面——西红柿鸡蛋卤面(窄面),你怎么加”   “那就从面条类里泛化一个西红柿鸡蛋卤面(窄面)类,然后再在它上面加上与西红柿鸡蛋卤和窄面的组合关系”   “那如果我现在面馆开10个分面馆,每个分面馆新增10样类,你是不是这100种面都要重新添加一遍面和卤的组合关系吧。”   “那我还不得累死啊,这种活干不得!”小B惶恐的说道。   “不管累,当你画出UML图的时候,能把你乱死。成品面与具体的面类型和卤菜类型的耦合性就太高了。想办法松散这些耦合。”   “那怎么办呢?”小B焦急的问道。   “呃,提醒你一点,你先对比一下你这两版UML各自的特点。”   “第一幅图继承关系比较明显,第二幅图则利用了组合关系,使其继承关系简单化了”   “如果你把这两个图合成一个图呢?”   “这,这,怎么合啊。”   “看来还得我出马啊,看着昂” UML图3.0版 ![](https://box.kancloud.cn/2016-01-14_56970cf24a891.png)   “不管什么打卤面,都抽象于卤面类。卤面类由面条类和卤菜类组成。面条类和卤菜类都有各自的子类。”   “这个图好面熟啊。让我想想昂。哦,对了,这不是桥接模式吗!”   “是的,不过这样做有什么好处呢?”   “这样做有几个好处,第一,松散耦合。具体的打卤面不再与具体的面类型和卤类型直接关联,松散了它们之间的耦合;第二,在变化方面,具体的打卤面, 具体的面类型和具体的卤类型各自的变化,互不影响”   “这样我在开分面馆的时候就不用画那么多关系了,添加新类型的打卤面不再成为难事。果然很厉害。哈哈哈哈”小B得瑟的笑着,好像他真的要开面馆似的。   “对头!这就满足了开闭原则,不用去修改,只需要添加即可。实例化具体的打卤面时,在客户端指定一下要哪种面,哪个卤即可。”小A接着说,    “其实这里面最重要的是利用聚合组合关系,松散了耦合,使得抽象不再依赖于具体,而具体要依赖于抽象。”   “哦,对哦。设计模式果然牛X。”   “代码留在晚上再写吧,早点休息,下午还有考试呢!”小A看着一脸丫丫的小B,提醒道。   “哦哦,差点忘了还有考试了。希望下午碰到桥接模式,那我就。。。”小B继续陷入丫丫ing。   小A摇摇头,不再理睬小B,推开霸占自己床铺的小C,休息去了。  面条类及子类 ~~~ #region 面条类及子类 /// <summary> /// 面条类 /// </summary> public abstract class Noodles { public Noodles() { } /// <summary> /// 获取面类型名称 /// </summary> /// <returns></returns> public abstract string GetName(); } /// <summary> /// 窄面条 /// </summary> public class NarrowNoodles:Noodles { public NarrowNoodles() { } public override string GetName() { return "窄面条"; } } /// <summary> /// 宽面条 /// </summary> public class WideNoodles : Noodles { public WideNoodles() { } public override string GetName() { return "宽面条"; } } #endregion ~~~ 卤菜类及子类 ~~~ #region 卤菜类及子类 /// <summary> /// 卤菜类 /// </summary> public abstract class Halogen { public Halogen() { } /// <summary> /// 获取卤菜名称 /// </summary> /// <returns></returns> public abstract string GetName(); } /// <summary> /// 西红柿鸡蛋卤 /// </summary> public class TomatoAndEgg : Halogen { public TomatoAndEgg() { } public override string GetName() { return "西红柿鸡蛋卤"; } } /// <summary> /// 茄子豆瓣酱 /// </summary> public class AubergineBeanPaste:Halogen { public AubergineBeanPaste() { } public override string GetName() { return "茄子豆瓣酱"; } } #endregion ~~~ 打卤面类及子类  ~~~ #region 打卤面类及子类 /// <summary> /// 打卤面类 /// </summary> public class NoodlesAndHalogen { protected string name; protected Noodles noodles; protected Halogen halogen; public NoodlesAndHalogen(string name, Noodles noodles, Halogen halogen) { this.name=name; this.noodles = noodles; this.halogen = halogen; } /// <summary> /// 获取名称及成分 /// </summary> public void GetName() { Console.WriteLine("我是"+name+",由"+noodles.GetName () +"和"+halogen.GetName ()+"组成"); } } /// <summary> /// 西红柿鸡蛋面 /// </summary> public class TomatoEggNoodles : NoodlesAndHalogen { public TomatoEggNoodles(string name, Noodles noodles, Halogen halogen) : base(name, noodles, halogen) { } } /// <summary> /// 苏格兰打卤面 /// </summary> public class ScotlandNoodles : NoodlesAndHalogen { public ScotlandNoodles(string name, Noodles noodles, Halogen halogen) : base(name, noodles,halogen) { } } #endregion ~~~ 客户端代码:  ~~~ static void Main(string[] args) { NoodlesAndHalogen n1=new TomatoEggNoodles("西红柿鸡蛋面(宽面)",new WideNoodles(),new TomatoAndEgg() ); n1.GetName(); NoodlesAndHalogen n2 = new ScotlandNoodles("苏格兰打卤面(窄面)", new NarrowNoodles(), new AubergineBeanPaste()); n2.GetName(); Console.Read(); } ~~~ 运行结果: ![](https://box.kancloud.cn/2016-01-14_56970cf25f28c.jpg) [![](https://box.kancloud.cn/2016-02-18_56c53c47e8b31.png)](http://my.csdn.net/my/favorite/miniadd?t=%25E9%259D%25A2%25E4%25B8%258E%25E5%258D%25A4%25E7%259A%2584%25E9%25B9%258A%25E6%25A1%25A5%25E7%259B%25B8%25E4%25BC%259A%25E2%2580%2594%25E2%2580%2594%25E6%25A1%25A5%25E6%258E%25A5%25E6%25A8%25A1%25E5%25BC%258F&u=http://blog.csdn.net/xiaoxian8023/article/details/7642352)