Net设计模式实例之抽象工厂模式(Abstract Factory Pattern)
一、抽象工厂模式简介(Bref Introduction)
抽象工厂模式(Abstract Factory Pattern),提供一个创建一系列相关或者相互依赖对象的接口,而无需制定他们的具体类。优点是:易于交换产品系列,由于具体工厂类在一个应该用中只需要在初始化的时候出现一次,这就使得改变一个应用的具体工厂类变得非常容易,它只需要改变具体工厂即可使用不同的产品配置。抽象工厂的另一个优点是,它让具体的创建实例与客户端分离,客户端是通过他们的抽象接口操纵实例,产品的具体类名也被具体工厂的实现分离,不会出现在客户的代码中。
二、解决的问题(What To Solve)
常用用于解决数据访问程序,也就是说程序通过使用抽象工厂模式后,可以灵活地在不同的数据库之间切换,而不需要费时费力地改变原有程序。
三、抽象工厂模式分析(Analysis)
1、抽象工厂模式结构
IProductA、IProductA接口:两个产品接口,他们都有可能有两种不同的实现。
ProductA1、ProductA2,ProductB1、ProductB2具体实现类:对两个产品接口的具体分类的实现。
AbstractFactory抽象类:抽象工厂接口,它里面应该包含所有产品创建的抽象方法。
ConcreteFactory1、ConcreteFactory2具体工厂类:创建具有特定实现的产品对象
2、源代码
|
1、产品接口IProductA、IProductB及其两种实现 |
|
public interface IProductA { void Show(); } public class ProductA1 : IProductA { public void Show() { Console.WriteLine("具体产品类{0}展示方法。",this.GetType().Name); } } public class ProductA2 : IProductA { public void Show() { Console.WriteLine("具体产品类{0}展示方法。", this.GetType().Name); } } public interface IProductB { void Insert(); } public class ProductB1 : IProductB { public void Insert() { Console.WriteLine("具体产品类{0}插入方法。", this.GetType().Name); } } public class ProductB2 : IProductB { public void Insert() { Console.WriteLine("具体产品类{0}插入方法。", this.GetType().Name); } } |
|
2、抽象工厂接口AbstractFactory,及其具体的工厂AbstractFactory1、AbstractFactory2 |
|
public abstract class AbstractFactory { public abstract IProductA CreateProductA(); public abstract IProductB CreateProductB(); } public class AbstractFactory1:AbstractFactory { public override IProductA CreateProductA() { IProductA productA1 = new ProductA1(); return productA1; } public override IProductB CreateProductB() { IProductB productB1 = new ProductB1(); return productB1; } } public class AbstractFactory2 : AbstractFactory { public override IProductA CreateProductA() { IProductA productA2 = new ProductA2(); return productA2; } public override IProductB CreateProductB() { IProductB productB2 = new ProductB2(); return productB2; } } |
|
3、客户端代码 |
|
static void Main(string[] args) { //根据需求调用具体工厂AbstractFactory1 AbstractFactory factory1 = new AbstractFactory1(); IProductA productA1 = factory1.CreateProductA(); IProductB productB1 = factory1.CreateProductB(); productA1.Show(); productB1.Insert(); Console.WriteLine("\n"); //根据需求调用具体工厂AbstractFactory2 AbstractFactory factory2 = new AbstractFactory2(); IProductA productA2 = factory2.CreateProductA(); IProductB productB2 = factory2.CreateProductB(); productA2.Show(); productB2.Insert(); Console.ReadKey(); } |
3、程序运行结果
四.案例分析(Example)
1、场景
使用抽象工厂+反射+配置文件实现数据访问层程序。结构如下图所示
用反射+抽象工厂+配置文件的数据访问程序。
Assembly.Load("程序集名称").CreateInstance("命名空间.类名称")。比如:
IProduct product=(IProduct)Assembly.Load("抽象工程模式").CreateInstance("抽象工程模式.SqlServerProduct")。
常用做法是:
Private static readonly string AssemblyName="抽象工程模式";
Private static readonly string DB=ConfiurationManager.AppSettings["db"];
配置文件如下:
<configuration>
<appSettings>
<add key="db" value="Sqlserver"/>
<appSettings>
<configuration>
通过读配置文件给DB字符赋值,在配置文件中写明当前使用的是SqlServer 还是Access数据库。反射+抽象工厂+配置文件解决方案解决了数据访问时的可维护、可扩展问题
2、代码
|
1、对象Uer、Product及其相对应的操作 |
|
public interface IUser { void Insert(); } public class SqlServerUser:IUser { public void Insert() { Console.WriteLine("{0}插入用户.",this.GetType().Name); } } public class AccessUser : IUser { public void Insert() { Console.WriteLine("{0}插入用户.", this.GetType().Name); } } public interface IProduct { void GetProduct(); } public class SqlServerProduct : IProduct { public void GetProduct() { Console.WriteLine("{0}查询商品.", this.GetType().Name); } } public class AccessProduct : IProduct { public void GetProduct() { Console.WriteLine("{0}查询商品.", this.GetType().Name); } } |
|
2、数据访问类DataAccess |
|
public class DataAccess { private static readonly string AssemblyName ="AbstractFactoryReflection"; private static readonly string db = "SqlServer"; public static IUser CreateUser() { string className = AssemblyName + "." + db + "User"; IUser user = (IUser)Assembly.Load(AssemblyName).CreateInstance(className); return user; } public static IProduct CreateProduct() { string className = AssemblyName + "." + db + "Product"; return(IProduct)Assembly.Load(AssemblyName).CreateInstance(className); } } |
|
3、客户端代码 |
|
static void Main(string[] args) { IUser user = DataAccess.CreateUser(); user.Insert(); IProduct product = DataAccess.CreateProduct(); product.GetProduct(); Console.ReadKey(); } |
五、总结(Summary)
抽象工厂模式(Abstract Factory Pattern),提供一个创建一系列相关或者相互依赖对象的接口,而无需制定他们的具体类。抽象工厂模式的典型应用就是,使用抽象工厂+反射+配置文件实现数据访问层程序。
转载自:http://www.cnblogs.com/ywqu/archive/2010/01/08/1641825.html
Net设计模式实例之抽象工厂模式(Abstract Factory Pattern)的更多相关文章
- 【设计模式】抽象工厂模式 Abstract Factory Pattern
简单工厂模式是一个工厂类根据工厂方法的参数创建不出不同的产品, 工厂方法模式是每一个产品都有一个一一对应的工厂负责创建该产品.那么今天要讲的抽象工厂模式是一个工厂能够产生关联的一系列产品.抽象工厂模式 ...
- 乐在其中设计模式(C#) - 抽象工厂模式(Abstract Factory Pattern)
原文:乐在其中设计模式(C#) - 抽象工厂模式(Abstract Factory Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 抽象工厂模式(Abstract Factor ...
- 设计模式 - 抽象工厂模式(abstract factory pattern) 具体解释
抽象工厂模式(abstract factory pattern) 详细解释 本文地址: http://blog.csdn.net/caroline_wendy/article/details/2709 ...
- 二十四种设计模式:抽象工厂模式(Abstract Factory Pattern)
抽象工厂模式(Abstract Factory Pattern) 介绍提供一个创建一系列相关或相互依赖对象的接口,而无需指定它们具体的类. 示例有Message和MessageModel,Messag ...
- 【UE4 设计模式】抽象工厂模式 Abstract Factory Pattern
概述 描述 提供一个创建一系列相关或相互依赖对象的接口,而无须指定它们具体的类:具体的工厂负责实现具体的产品实例 抽象工厂中每个工厂可以创建多种产品(如苹果公司生产iPhone.iPad): 工厂方法 ...
- 设计模式学习心得<抽象工厂模式 Abstract Factory>
抽象工厂模式(Abstract Factory Pattern)是围绕一个超级工厂创建其他工厂.该超级工厂又称为其他工厂的工厂.这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式. 在抽 ...
- IOS设计模式浅析之抽象工厂模式(Abstract Factory)
概述 在前面两章中,分别介绍了简单工厂模式和工厂方法模式,我们知道简单工厂模式的优点是去除了客户端与具体产品的依赖,缺点是违反了“开放-关闭原则”:工厂方法模式克服了简单工厂模式的缺点,将产品的创建工 ...
- 设计模式之抽象工厂模式(Abstract Factory Pattern)
一.抽象工厂模式的由来 抽象工厂模式,最开始是为了解决操作系统按钮和窗体风格,而产生的一种设计模式.例如:在windows系统中,我们要用windows设定的按钮和窗体,当我们切换Linux系统时,要 ...
- C#设计模式——抽象工厂模式(Abstract Factory Pattern)
一.概述在软件开发中,常常会需要创建一系列相互依赖的对象,同时,由于需求的变化,往往存在较多系列对象的创建工作.如果采用常规的创建方法(new),会造成客户程序和对象创建工作的紧耦合.对此,抽象工厂模 ...
随机推荐
- c++ eof()函数
C++ eof()函数可以帮助我们用来判断文件是否为空,抑或是判断其是否读到文件结尾.在这里我们将会对其进行详细的介绍. C++编程语言中的很多功能在我们的实际应用中起着非常大的作用.比如在对文件文本 ...
- 谷歌发布的首款基于HTTP/2和protobuf的RPC框架:GRPC
Google 刚刚开源了grpc, 一个基于HTTP2 和 Protobuf 的高性能.开源.通用的RPC框架.Protobuf 本身虽然提供了RPC 的定义语法,但是一直以来,Google 只开 ...
- STOMP协议介绍
STOMP,Streaming Text Orientated Message Protocol,是流文本定向消息协议,是一种为MOM(Message Oriented Middleware,面向消息 ...
- 一个基于Orchard的开源CRM --coevery简介
Coevery是开源的.NET Web平台项目,力争打造一个开放而鲁棒的CRM系统,采用Orchard架构,并使用AngularJS改善页面体验.作为一个后发优势的CRM 产品,Coevery 具有一 ...
- Kosaraju 算法检测有向图的强连通性
给定一个有向图 G = (V, E) ,对于任意一对顶点 u 和 v,有 u --> v 和 v --> u,亦即,顶点 u 和 v 是互相可达的,则说明该图 G 是强连通的(Strong ...
- CSS尺寸和字体单位-em、px还是%
在页面整体布局中,页面元素的尺寸大小(长度.宽度.内外边距等)和页面字体的大小也是重要的工作之一.一个合理设置,则会让页面看起来层次分明,重点鲜明,赏心悦目.反之,一个不友好的页面尺寸和字体大小设置, ...
- MySQL外键之级联
简介 MySQL外键起到约束作用,在数据库层面保证数据的完整性.例如使用外键的CASCADE类型,当子表(例如user_info)关联父表(例如user)时,父表更新或删除时,子表会更新或删除记录,这 ...
- 修改windows自带的Ctrl+Space输入法切换快捷键
使用场景: 多为我等码农使用一些编辑器时,编辑器的默认代码提示热键为 ctrl+space ,但这个热键被系统的输入法开关占用.如果遇到可以设置快捷键的编辑器还好,要是不能设置的话(比如火狐浏览器的代 ...
- maven的聚合与继承
新建一个空的maven项目user-parent Pom.xml内容 <project xmlns="http://maven.apache.org/POM/4.0.0" x ...
- 分区函数Partition By的与row_number()的用法以及与排序rank()的用法详解(获取分组(分区)中前几条记录)
partition by关键字是分析性函数的一部分,它和聚合函数不同的地方在于它能返回一个分组中的多条记录,而聚合函数一般只有一条反映统计值的记录,partition by用于给结果集分组,如果没有指 ...