《深入浅出设计模式-中文版》读书笔记 开篇乱弹(一)

  oreilly的《Head.First设计模式》,《深入浅出设计模式》是一本好书,用生动的语言讲解设计模式。而且是逐级深入,并没有一上来就gof23,就一大套设计模式的理论。而是用例子切入,使得读者逐步的理解设计模式的好处,理解设计模式可以解决的问题,使我们有了继续读下去的理由和勇气,而不至于被大片的理论所击倒。

  更加容易入门,为以后读那些理论打下坚实的基础。

  感谢作者写出一本这么好的书。

  设计原则一:

  找出应用中可能需要变化的地方,把他们独立出来,不要和那些不需要变化的代码混合在一起。

  把会变化的部分取出来“封装”起来,以便以后可以轻易的修改和扩展此部分,而不会影响不需要变化的部分。

  单一职责原则。

  设计原则二:

  针对接口编程,而不是针对实现编程。

  针对接口编程,这里的接口是广义的接口,不止包括语法中的接口interface

  

<!--<br/ /><br/ />Code highlighting produced by Actipro CodeHighlighter (freeware)<br/ />http://www.CodeHighlighter.com/<br/ /><br/ />--> public   interface  IFlyBehavior
   {
       
void  Fly ();
   }
  

 

  还包括抽象类,其实就是一个超类型,反正不是具体的实现类,这样就可以利用多态,不会被绑死到一种具体的类型和实现上。这句话也可以解释为变量的声明应该是超类型,通常是一个抽象类或者是一个接口,因此只要是实现了超类的具体类产生的对象,都可以赋值给这个变量。

  

代码
<!--<br/ /><br/ />Code highlighting produced by Actipro CodeHighlighter (freeware)<br/ />http://www.CodeHighlighter.com/<br/ /><br/ />--> public   abstract   class  Animal
    {
        
public   abstract   void  MakeSound();
    }
    
public   class  Dog : Animal
    {
        
public   override   void  MakeSound()
        {
            bark();
        }
        
public   void  bark()
        { }
    }
    
public   class  Cat : Animal
    {
        
public   override   void  MakeSound()
        {
            meow();
        }
        
public   void  meow()
        { }
    }

 

  针对实现编程的调用代码

  

<!--<br/ /><br/ />Code highlighting produced by Actipro CodeHighlighter (freeware)<br/ />http://www.CodeHighlighter.com/<br/ /><br/ />-->   public   void  Caller1()
        {
            Dog dog 
=   new  Dog();
            dog.bark();
        }
  

 

  针对接口编程的调用代码

  

<!--<br/ /><br/ />Code highlighting produced by Actipro CodeHighlighter (freeware)<br/ />http://www.CodeHighlighter.com/<br/ /><br/ />-->   public   void  Caller2()
        {
            Animal a 
=   new  Cat();
            a.MakeSound();
        }

 

  在上面的代码中Animal a = new Cat();动物的创建还是被硬编码了,我们还可以在代码运行的时候再来指定创建的类型,就是我们不关心创建的类型是什么,只要可以进行makesound就可以了。方法有很多,常见的就是利用一个工厂类,将对象的创建交给工程类来完成,只要告诉它需要创建的对象的特征即可。这里我们简单的使用对象的类型名称代表对象。

  

  简单工厂类

代码
<!--<br/ /><br/ />Code highlighting produced by Actipro CodeHighlighter (freeware)<br/ />http://www.CodeHighlighter.com/<br/ /><br/ />--> public   class  AnimalFactory
    {
        
public   static  Animal CreateAnimal( string  type)
        {
            
switch  (type)
            {
                
case   " dog " :
                    
return   new  Dog();
                    
break ;
                
case   " cat " :
                    
return   new  Cat();
                    
break ;
                
default :
                    
return   null ;
                    
break ;
            }
        }
    }

 

    
  调用方

  

<!--<br/ /><br/ />Code highlighting produced by Actipro CodeHighlighter (freeware)<br/ />http://www.CodeHighlighter.com/<br/ /><br/ />-->   public   void  Caller3()
        {
            Animal a 
=  AnimalFactory .CreateAnimal ( " cat " );
            a.MakeSound();
        }
  

 

  但是我们还是在工厂方法中看到了switch或者是很多的if。。。else。。。,如果有新类型添加,这个静态方法还是要不断的添加,越来越长,就会不符合OPC原则,有什么更好的办法吗?

  答案是:有。

  我们可以将系统现有的类型都预先创建出来,然后再外部调用静态方法的时候,给他需要的对象就可以了。

  

代码
<!--<br/ /><br/ />Code highlighting produced by Actipro CodeHighlighter (freeware)<br/ />http://www.CodeHighlighter.com/<br/ /><br/ />-->   public   class  AnimalFactory
    {
        
private  Dictionary < string , Animal >  _animals;
        
private  AnimalFactory()
        {
            _animals 
=   new  Dictionary < string , Animal > ();
            _animals.Add(
" dog " new  Dog());
            _animals.Add(
" cat " new  Cat());
        }
        
private   static  AnimalFactory _instance  =   new  AnimalFactory();
        
private   static    object  _lockObj  =   new   object ();
        
public   static  AnimalFactory GetInstance()
        {
            
if  (_instance  ==   null )
            {
                
lock  (_lockObj)
                {
                    
if  (_instance  ==   null )
                        _instance 
=   new  AnimalFactory();
                }
            }
            
return  _instance;
        }

        
public   Animal CreateAnimal( string  type)
        {
            
return  _animals[type];
            
// switch (type)
            
// {
            
//     case "dog":
            
//         return new Dog();
            
//         break;
            
//     case "cat":
            
//         return new Cat();
            
//         break;
            
//     default:
            
//         return null;
            
//         break;
            
// }
        }
    }

  调用方代码

  

 

<!--<br/ /><br/ />Code highlighting produced by Actipro CodeHighlighter (freeware)<br/ />http://www.CodeHighlighter.com/<br/ /><br/ />-->   public   void  Caller3()
        {
            Animal a 
=  AnimalFactory.GetInstance () .CreateAnimal ( " cat " );
            a.MakeSound();
        }

 

  我们先创建一个字典,作为容器,预先放置了全部的类型及其类型名称,然后再需要的时候取一个出来就可以了。如果有新类型添加,也不用每次都添加一个if或者是case了,只要修改工厂方法的静态构造函数,在字典中添加新元素就可以了。

  这时候有些人会觉得这里好像很眼熟,对了,就是容器,IOC容器,依赖注入,我们不知不觉的重构到了IOC容器。IOC容器就是一个类似字典的东西,使得你事先可以注册很多的类型在里面,设置好这些类型的映射关系,配置好容器之后,容器中就会有很多的类型实例,然后再用容器的Resolv方法来获取指定类型的实例,而不用new方法来创建了。

  而且上面最后的工厂类,我们还使用了单件模式以及简单工厂模式。

  类型的注册可以采用代码编写,就好像向字典中添加元素一样。也可以采用xml配置文件,更加灵活。

  

你可能感兴趣的:(设计模式,编程,IOC,读书)