在工厂模式家族中最出名的是工厂三姐妹,根据抽象的程度不同分为简单工厂、工厂模式和抽象工厂模式。他们在我们平时的编程中会经常使用。所以我们应该详细地了解一下他们三者之间优缺点。
简单工厂
定义
简单工厂模式又叫做静态工厂方法(Static FactoryMethod)模式,但不属于23种GOF设计模式之一。简单工厂模式是由一个工厂对象决定创建出哪一种产品类的实例。简单工厂模式是工厂模式家族中最简单实用的模式,可以理解为是不同工厂模式的一个特殊实现。
举例说明:
代码实现:
//客户端代码 static void Main(string[] args) { Operation oper; oper = OperationFactory.createOperate("*"); oper.NumberA = 1; oper.NumberB = 2; double result = oper.GetResult(); } } //Operation运算类 public class Operation { private double _numberA = 0; private double _numberB = 0; public double NumberA { get { return _numberA; } set { _numberA = value; } } public double NumberB { get { return _numberB; } set { _numberB = value; } } public virtual double GetResult() { double result = 0; return result; } } //加减乘除类 class OperationAdd : Operation //加法类,继承运算类 { public override double GetResult() { double result = 0; result = NumberA + NumberB; return result; } } class OperationSub : Operation //减法类,继承运算类 { public override double GetResult() { double result = 0; result = NumberA - NumberB; return result; } } class OperationMul : Operation //乘法类,继承运算类 { public override double GetResult() { double result = 0; result = NumberA * NumberB; return result; } } class OperationDiv : Operation //除法类,继承运算类 { public override double GetResult() { double result = 0; if (NumberB == 0) throw new Exception("除数不能为0。"); result = NumberA / NumberB; return result; } } //简单运算工厂类 class OperationFactory { public static Operation createOperate(string operate) { Operation oper = null; switch (operate) { case "+": oper = new OperationAdd(); break; case "-": oper = new OperationSub(); break; case "*": oper = new OperationMul(); break; case "/": oper = new OperationDiv(); break; } return oper; } }
简单工厂的优点:
1、工厂类是整个模式的关键.包含了必要的逻辑判断,根据给定的信息,决定究竟应该创建哪个具体类的实例。
2、明确了各自的职责和权利,有利于整个软件体系结构的优化。
简单工厂的缺点:
1、由于工厂类集中了所有实例的创建逻辑,违反了高内聚低耦合原则,将全部创建逻辑集中到了一个工厂类中;
2、它所能创建的类只能是事先考虑到的,每次扩展都需要更改工厂类,违反了开放封闭原则。
简单工厂模式的实质是由一个工厂类根据传入的参数,动态决定应该创建哪一个产品类的实例。
工厂模式
定义:
用于创建对象的接口,让子类决定实例化哪一个类。工厂模式对类的实例化延迟到其子类。
举例说明:数据访问程序
代码实现:
static void Main(string[] args) { User user = new User(); IFactory factory = new SqlServerFactory(); IUser iu = factory.CreateUser(); iu.Insert(user); iu.GetUser(1); Console.Read(); } } //User类 class User { private int _id; public int ID { get {return _id;} set {_id=value;} } private string _name; public string Name { get {return _name;} set {_name =value ;} } } //IUser接口 interface IUser { void Insert(User user); User GetUser(int id); } //SqlserverUser类 class SqlserverUser : IUser { public void Insert(User user) { Console.WriteLine("在SQL Server 中给User表增加一条记录"); } public User GetUser(int id) { Console.WriteLine("在SQL Server 中根据ID得到User表一条记录"); return null; } } //AccessUser类 class AccessUser : IUser { public void Insert(User user) { Console.WriteLine("在Access中给User表增加一条记录"); } public User GetUser(int id) { Console.WriteLine("在Access中根据ID得到User表一条记录"); return null; } } //IFactory接口 interface IFactory { IUser CreateUser(); } //SqlServerFactory :IFactory class SqlServerFactory : IFactory { public IUser CreateUser() { return new AccessUser(); } } //AccessFactory类 class AccessFactory:IFactory { public IUser CreateUser() { return new AccessUser(); } }
工厂模式的优点:
1、克服了简单工厂违背开放封闭原则的缺点,保持了封装的优点。工厂模式集中封装了对象的创建,更换对象时更加容易。 2、使用了多态性,保持了简单工厂的优点。 3、降低耦合度。降低了客户程序与产品对象的耦合。抽象工厂
定义:
创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类。举例说明:数据访问程序
代码实现:
static void Main(string[] args) { User user=new User (); Department dept = new Department(); //IFactory factory = new SqlServerFactory(); IFactory factory = new AccessFactory(); IUser iu = factory.CreateUser(); iu.Insert(user); iu.GetUser(1); IDepartment id = factory.CreateDepartment(); id.Insert(dept); id.GetDepartment(1); Console.Read(); } } class User //用户类 { private int _id; public int ID { get {return _id;} set {_id =value;} } private string _name; public string Name { get {return _name ;} set {_name =value ;} } } //IUser接口 interface IUser { void Insert(User user); User GetUser(int id); } // IUser接口的子SqlserverUser类,用于访问SQL Server的User class SqlserverUser :IUser { public void Insert(User user) { Console.WriteLine("在SQL Server中给User表增加一条记录"); } public User GetUser(int id) { Console.WriteLine ("在SQL Server中根据ID得到User表一条记录"); return null; } } // IUser接口的子AccessUser类,用于访问Access的User class AccessUser :IUser { public void Insert(User user) { Console.WriteLine ("在Access中给User表增加一条记录"); } public User GetUser(int id) { Console.WriteLine ("在Access中根据ID得到User表中一条记录"); return null; } } //增加一个Department表 class Department { private int _id; public int ID { get { return _id; } set { ID = value; } } private string _deptName; public string DeptName { get { return _deptName; } set { DeptName = value; } } } //IDpartment接口 interface IDepartment { void Insert(Department department); Department GetDepartment(int id); } //sqlserverDepartment类 class SqlserverDepartment : IDepartment { public void Insert(Department department) { Console.WriteLine("在SQL Server中给Department表增加一条记录"); } public Department GetDepartment(int id) { Console.WriteLine("在SQL Server中根据ID得到Department表一条记录"); return null; } } //AccessDepartment类 class AccessDepartment : IDepartment { public void Insert(Department department) { Console.WriteLine("在Access中给Departmen表增加一条记录"); } public Department GetDepartment(int id) { Console.WriteLine("在Access中根据ID得到Departmen表一条记录"); return null; } } //抽象工厂的接口 interface IFactory { IUser CreateUser(); IDepartment CreateDepartment(); //增加的接口方法 } class SqlServerFactory : IFactory { public IUser CreateUser() { return new SqlserverUser(); } public IDepartment CreateDepartment() { return new AccessDepartment(); } } class AccessFactory : IFactory { public IUser CreateUser() { return new AccessUser(); } public IDepartment CreateDepartment() { return new AccessDepartment(); } }
抽象工厂的优点:
抽象工厂模式除了具有工厂方法模式的优点外,最主要的优点就是可以在类的内部创建不同的产品对象。抽象工厂的缺点:
对于增加需求,扩展程序时,需要增加许多子类,使得程序变得臃肿,麻烦。(此时可以利用放射+抽象工厂的方法避免)。
总结:
无论是简单工厂模式,工厂模式,还是抽象工厂模式,他们都属于工厂模式,并且是创建型模式,在形式和特点上也是极为相似的,他们的最终目的都是为了解耦。