一、前言

工厂模式常见的词:简单工厂、工厂方法、抽象工厂。简单工厂不属于23种经典设计模式,但通常将它作为学习其他工厂模式的基础。

二、简单工厂1、定义

定义一个工厂类,它可以根据参数的不同返回不同类型的实例,被创建的实例通常都具有共同的父类。由于简单的工厂模式中用于创建实例的方法通常是静态方法,所以简单工厂又被称作为静态工厂。

2、例子

例:开发一套图表库,根据不同的参数可以创建柱状图BarChart,直线图LineChart,饼状图PieChart,下面使用简单工厂来实现。
Chart:抽象图表接口,充当抽象产品类

namespace _02_FactoryMethod{    internal interface Chart    {        void Display();    }}

BarChart:柱状图类,充当具体产品类

namespace _02_FactoryMethod{    class BarChart : Chart    {        public BarChart()        {            Console.WriteLine("创建柱状图");        }        public void Display()        {            Console.WriteLine("显示柱状图");        }    }}

LineChart:折线图类,充当具体产品类

namespace _02_FactoryMethod{    internal class LineChart : Chart    {        public LineChart()        {            Console.WriteLine("创建折线图");        }        public void Display()        {            Console.WriteLine("显示折线图");        }    }}

PieChart:饼图类,充当具体产品类

namespace _02_FactoryMethod{    internal class PieChart : Chart    {        public PieChart()        {            Console.WriteLine("创建饼图");        }        public void Display()        {            Console.WriteLine("显示饼图");        }    }}

ChartFactory:图表工厂类,充当工厂类

namespace _02_FactoryMethod{    internal class ChartFactory    {        public static Chart GetChart(string type)        {            Chart chart = null;            if (type.Equals("bar"))            {                chart = new BarChart();            }            else if (type.Equals("line"))            {                chart = new LineChart();            }            else if (type.Equals("pie"))            {                chart = new PieChart();            }            return chart;        }    }}

Program:测试代码

Chart chart = ChartFactory.GetChart("bar");chart.Display();

简单工厂也可以再简化,直接将抽象类和工厂类合并,因为并不是所有类型的产品都那么复杂。

3、总结

实际应用中,简单工厂肯定是难当大任的,工厂类职责过重,一旦出现问题,整个程序直接崩溃,而且产品类不易拓展,新增就意味着改代码。当然,如果只是一个简单程序,配合使用反射、配置文件等手段可以快速完成开发,相信大家都“吃过猪肉也见过猪跑”。

三、工厂方法1、定义

定义一个用于创建对象的接口,但是让子类决定将哪一个实例化。工厂方法模式让一个类的实例化延迟到其子类,是一种创建型模式。与简单工厂相比,引入了抽象工厂角色。工厂方法模式包含以下四个角色:

1、Product(抽象产品):它是定义产品的接口,是工厂方法模式所创建对象的超类型,也就是产品对象的公共父类。
2、ConcreteProduct(具体产品):它实现了抽象产品接口,某种类型的具体产品由专门的具体工厂创建,具体工厂和具体产品之间一一对应。
3、Factory(抽象工厂):在抽象工厂类中,声明了工厂方法(Factory Method),用于返回一个产品。抽象工厂是工厂方法模式的核心,所有创建对象的工厂类都必须实现该接口。
4、ConcreteFactory(具体工厂):它是抽象工厂类的子类,实现了在抽象工厂中声明的工厂方法,由客户端调用,返回一个具体产品类的实例。

2、例子

例:系统封装多种记录日志方式,包括文件记录日志、数据库记录日志等,可以根据需求切换。
LoggerFactory:抽象产品工厂接口,供具体产品(数据库记录日志、文件记录日志)工厂继承,实现创建具体产品类的方法

public interface LoggerFactory{    public Logger CreateLogger();}

DataBaseLoggerFactory:数据库记录日志工厂类,继承抽象产品工厂接口

public class DataBaseLoggerFactory : LoggerFactory{    public Logger CreateLogger()    {        //创建数据库业务代码略        Logger logger = new DataBaseLogger();        return logger;    }}

FileLoggerFactory:文件记录日志工厂类,继承抽象产品工厂接口

public class FileLoggerFactory : LoggerFactory{    public Logger CreateLogger()    {        //创建文件业务代码略        Logger logger = new FileLogger();        return logger;    }}

Logger:抽象产品接口,供具体产品类继承,实现记录日志的具体方法

public interface Logger{    void WriteLog();}

DataBaseLogger:数据库记录日志类,继承抽象产品

public class DataBaseLogger : Logger{    public void WriteLog()    {        Console.WriteLine("数据库记录日志");    }}

FileLogger:文件记录日志类,继承抽象产品

public class FileLogger : Logger{    public void WriteLog()    {        Console.WriteLine("文件记录日志");    }}

Program:客户端测试类

LoggerFactory loggerFactory = new DataBaseLoggerFactory();Logger logger = loggerFactory.CreateLogger();logger.WriteLog();

3、总结

工厂方法模式除了包含简单工厂的优点,还弥补了不足,当需要添加新产品时,只需要添加一个具体工厂和具体产品即可,无需改动已有代码,增强系统可扩展性,符合开闭原则。当然这同时也增加了系统的理解难度,具体产品类过多的话,系统将变得无比庞大。

测试签名