zoukankan      html  css  js  c++  java
  • 设计模式:简单工厂、工厂方法、抽象工厂之小结与区别 (转)

    简单工厂,工厂方法,抽象工厂都属于设计模式中的创建型模式。其主要功能都是帮助我们把对象的实例化部分抽取了出来,优化了系统的架构,并且增强了系统的扩展性。

    本文是本人对这三种模式学习后的一个小结以及对他们之间的区别的理解。

    简单工厂

    简单工厂模式的工厂类一般是使用静态方法,通过接收的参数的不同来返回不同的对象实例。

    不修改代码的话,是无法扩展的。

    简单工厂

    工厂方法

    工厂方法是针对每一种产品提供一个工厂类。通过不同的工厂实例来创建不同的产品实例。

    在同一等级结构中,支持增加任意产品。

    工厂方法

    抽象工厂

    抽象工厂是应对产品族概念的。比如说,每个汽车公司可能要同时生产轿车,货车,客车,那么每一个工厂都要有创建轿车,货车和客车的方法。

    应对产品族概念而生,增加新的产品线很容易,但是无法增加新的产品。

    抽象工厂

    小结

    ★工厂模式中,重要的是工厂类,而不是产品类。产品类可以是多种形式,多层继承或者是单个类都是可以的。但要明确的,工厂模式的接口只会返回一种类型的实例,这是在设计产品类的时候需要注意的,最好是有父类或者共同实现的接口。

    ★使用工厂模式,返回的实例一定是工厂创建的,而不是从其他对象中获取的。

    ★工厂模式返回的实例可以不是新创建的,返回由工厂创建好的实例也是可以的。

    区别

    简单工厂 用来生产同一等级结构中的任意产品。(对于增加新的产品,无能为力)

    工厂模式 :用来生产同一等级结构中的固定产品。(支持增加任意产品)  
    抽象工厂:用来生产不同产品族的全部产品。(对于增加新的产品,无能为力;支持增加产品族)  

    以上三种工厂 方法在等级结构和产品族这两个方向上的支持程度不同。所以要根据情况考虑应该使用哪种方法。  

    -----------------------------------------------------------------------------------------------------------------------------------------------------------------------

    代码示例:

    看了网络上很多关于设计模式的方法,有的模式看起来相似,但本质还是区别很大的.像简单工厂,工厂方法和抽象工厂就有很明显的区别.
      有一个比较好理解的例子,在这跟大家介绍下:
      1.如果一个后花园只种蔬菜类,那么就用简单工厂就可以了.
      2.如果后花园蔬菜品种繁多.得用工厂方法才可以,把共有的东西抽象出来.
      3.如果要扩大后花园的规模,比如一个在北方,一个在南方,这样工厂方法就无法实现了,就应当用抽象工厂,把各种各样的植物,又组成一个后花园.
    所以我个人认为,简单工厂是一个工厂只生产一类的产品,面对的是具体的类,工厂方法是可以生产不同的产品,把公共的方法抽象出来,然后进行创建各种各样的产品.抽象工厂把几种产品划出共同的东西,把相互依赖的对象抽象出来,只要实现这些接口就可以得到不同的产品.
    具体例子: 
     1.简单工厂:
    using System;
    public interface ICar
    {
        void run();
    }

    public class BMWCar : ICar
    {
        public void run()
        {
            Console.WriteLine("BMWCar run");
        }
    }

    public class BenzCar : ICar
    {
        public void run()
        {
            Console.WriteLine("BenzCar run");
        }
    }

    public class Driver
    {
        public static ICar DriverCar(string carType)
        {
            switch (carType)
            {
                case "BMWCar": return new BMWCar();
                case "BenzCar": return new BenzCar();
                default: throw new Exception();
           }
        }
    }

    public class Client
    {
        public static void Main()
        {
            ICar myCar = Driver.DriverCar("BenzCar");
            myCar.run();
            Console.Read();
        }
    }

    心得:优点是只要实现共有的接口就可以实现不同车跑的方式.但缺点就是要判断哪一种车,造成要修改Driver 类
    2.工厂方法:

    using System; 
    public interface ICar 
    { 
        void run(); 
    } 
    
    public class BMWCar : ICar 
    { 
        public void run() 
        { 
            Console.WriteLine("BMWCar run"); 
        } 
    } 
    
    public class BenzCar : ICar 
    { 
        public void run() 
        { 
            Console.WriteLine("BenzCar run"); 
        } 
    } 
    public abstract class Driver 
    { 
        public abstract ICar DriverCar(); 
    } 
    
    public class BMWDriver : Driver 
    { 
        public override ICar  DriverCar() 
        { 
            return new BMWCar(); 
        } 
    
    } 
    public class BenzDriver : Driver 
    { 
        public  override ICar DriverCar() 
        { 
            return new BenzCar(); 
        } 
    } 
    
    class Client 
    { 
        public static void Main() 
        { 
            Driver myDriver = new BenzDriver(); 
            ICar myCar = myDriver.DriverCar(); 
            myCar.run(); 
            Console.Read(); 
        } 
    }

    心得:优点是符合了开放-封闭原则(OCP),从整体上还看不出什么缺点.

    3.抽象工厂:

    using System; 
    
    public interface IBusinessCar 
    { 
        void run(); 
    } 
    
    public interface ISportCar 
    { 
        void run(); 
    } 
    
    public class BMWBusinessCar : IBusinessCar 
    { 
        public void run() 
        { 
            Console.WriteLine("BMWCar run"); 
        }
    }
    
    public class BenzBusinessCar : IBusinessCar 
    { 
        public void run() 
        { 
            Console.WriteLine("BenzBusinessCar run"); 
        } 
    } 
    
    public class BMWSportCar:ISportCar 
    { 
        public void run() 
        { 
            Console.WriteLine("BMWSportCar run"); 
        } 
    } 
    
    public class BenzSportCar:ISportCar 
    { 
        public void run() 
        { 
            Console.WriteLine("BenzSportCar run"); 
        } 
    } 
    
    public interface IDriver 
    { 
        IBusinessCar BusinessCarDriver(); 
        ISportCar SportCarDriver(); 
    } 
    
    public class BMWDriver:IDriver 
    { 
        public IBusinessCar BusinessCarDriver() 
        { 
            return new BMWBusinessCar(); 
        } 
        public ISportCar SportCarDriver() 
        { 
            return new BMWSportCar(); 
        } 
    } 
    
    public class BenzDriver:IDriver 
    { 
        public IBusinessCar BusinessCarDriver() 
        { 
             return new BenzBusinessCar(); 
        } 
    
        public ISportCar SportCarDriver() 
        { 
            return new BenzSportCar(); 
        } 
    }
    
    class Client 
    { 
        public static void Main() 
        { 
            IDriver myDriver =new BenzDriver(); 
            ISportCar myCar = myDriver.SportCarDriver(); 
            myCar.run(); 
            Console.Read(); 
        }
    
    }

    心得:抽象方法似乎达到了完美境界.把开奔驰的司机和开宝马的司机的公共方法抽象出来,并对不同的司机创建不同的类,到时候不管是开什么车的司机随你添加.它们唯一的共同点都是开车.

  • 相关阅读:
    Tomcat 结合Atomikos 实现JTA
    [译]Kinect for Windows SDK开发入门(十一):手势识别 下:基本手势识别
    [译]Kinect for Windows SDK开发入门(十二):语音识别 上
    [译]Kinect for Windows SDK开发入门(十):手势识别 上:基本概念
    [译]Kinect for Windows SDK开发入门(十五):进阶指引 下
    [译]Kinect for Windows SDK开发入门(九):骨骼追踪进阶 下
    [译]Kinect for Windows SDK开发入门(十四):进阶指引 上
    [译]Kinect for Windows SDK开发入门(八):骨骼追踪进阶 上
    [译]Kinect for Windows SDK开发入门(十三):语音识别 下
    Codeforces Round #757 (Div. 2) A D
  • 原文地址:https://www.cnblogs.com/icedlemon/p/4300583.html
Copyright © 2011-2022 走看看