**本文来自:崔成龙博客专栏。转载请注明出处:**[**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)