Net设计模式实例之抽象工厂模式(Abstract Factory Pattern)

本文涉及的产品
RDS SQL Server Serverless,2-4RCU 50GB 3个月
推荐场景:
云数据库 RDS SQL Server,基础系列 2核4GB
简介: 一、抽象工厂模式简介(Bref Introduction) 抽象工厂模式(Abstract Factory Pattern),提供一个创建一系列相关或者相互依赖对象的接口,而无需制定他们的具体类。

一、抽象工厂模式简介(Bref Introduction

抽象工厂模式(Abstract Factory Pattern),提供一个创建一系列相关或者相互依赖对象的接口,而无需制定他们的具体类。优点是:易于交换产品系列,由于具体工厂类在一个应该用中只需要在初始化的时候出现一次,这就使得改变一个应用的具体工厂类变得非常容易,它只需要改变具体工厂即可使用不同的产品配置。抽象工厂的另一个优点是,它让具体的创建实例与客户端分离,客户端是通过他们的抽象接口操纵实例,产品的具体类名也被具体工厂的实现分离,不会出现在客户的代码中。

二、解决的问题(What To Solve

       常用用于解决数据访问程序,也就是说程序通过使用抽象工厂模式后,可以灵活地在不同的数据库之间切换,而不需要费时费力地改变原有程序。

三、抽象工厂模式分析(Analysis

1、抽象工厂模式结构

 

IProductAIProductA接口:两个产品接口,他们都有可能有两种不同的实现。

ProductA1ProductA2ProductB1ProductB2具体实现类:对两个产品接口的具体分类的实现。

AbstractFactory抽象类:抽象工厂接口,它里面应该包含所有产品创建的抽象方法。

ConcreteFactory1ConcreteFactory2具体工厂类:创建具有特定实现的产品对象

2、源代码

1、产品接口IProductAIProductB及其两种实现

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,及其具体的工厂AbstractFactory1AbstractFactory2

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、对象UerProduct及其相对应的操作

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

如果你认为此文章有用,请点击底端的【推荐】让其他人也了解此文章,

img_2c313bac282354945ea179a807d7e70d.jpg

本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。

 

相关实践学习
使用SQL语句管理索引
本次实验主要介绍如何在RDS-SQLServer数据库中,使用SQL语句管理索引。
SQL Server on Linux入门教程
SQL Server数据库一直只提供Windows下的版本。2016年微软宣布推出可运行在Linux系统下的SQL Server数据库,该版本目前还是早期预览版本。本课程主要介绍SQLServer On Linux的基本知识。 相关的阿里云产品:云数据库RDS&nbsp;SQL Server版 RDS SQL Server不仅拥有高可用架构和任意时间点的数据恢复功能,强力支撑各种企业应用,同时也包含了微软的License费用,减少额外支出。 了解产品详情:&nbsp;https://www.aliyun.com/product/rds/sqlserver
相关文章
|
Java C#
使用C# (.NET Core) 实现适配器模式 (Adapter Pattern) 和外观模式 (Facade Pattern)
本文的概念内容来自深入浅出设计模式一书 现实世界中的适配器(模式) 我带着一个国标插头的笔记本电脑, 来到欧洲, 想插入到欧洲标准的墙壁插座里面, 就需要用中间这个电源适配器. 面向对象的适配器 你有个老系统, 现在来了个新供应商的类, 但是它们的接口不同, 如何使用这个新供应商的类呢? 首先, 我们不想修改现有代码, 你也不能修改供应商的代码.
1773 0
|
C# 设计模式 .NET
使用C# (.NET Core) 实现状态设计模式 (State Pattern)
本文的概念性内容来自深入浅出设计模式一书 项目需求 这是一个糖果机的需求图.  它有四种状态, 分别是图中的四个圆圈: No Quarter: 无硬币 Has Quater 有硬币 Gumball Sold 糖果卖出 Out of Gumball 没有糖果了 这个图很像一个状态图.
1887 0
|
C#
使用C# (.NET Core) 实现组合设计模式 (Composite Pattern)
本文的概念性内容来自深入浅出设计模式一书. 本文需结合上一篇文章(使用C# (.NET Core) 实现迭代器设计模式)一起看. 上一篇文章我们研究了多个菜单一起使用的问题. 需求变更 就当我们感觉我们的设计已经足够好的时候, 新的需求来了, 我们不仅要支持多种菜单, 还要支持菜单下可以拥有子菜单.
1464 0
|
Java C# 设计模式
使用C# (.NET Core) 实现迭代器设计模式 (Iterator Pattern)
本文的概念来自深入浅出设计模式一书 项目需求 有两个饭店合并了, 它们各自有自己的菜单. 饭店合并之后要保留这两份菜单. 这两个菜单是这样的: 菜单项MenuItem的代码是这样的: 最初我们是这样设计的, 这是第一份菜单: 这是第2份菜单: 同时有两个菜单存在的问题 问题就是多个菜单把事情变复杂了.
1036 0
|
算法 C# Java
使用C# (.NET Core) 实现模板方法模式 (Template Method Pattern)
本文的概念内容来自深入浅出设计模式一书. 项目需求 有一家咖啡店, 供应咖啡和茶, 它们的工序如下: 咖啡: 茶: 可以看到咖啡和茶的制作工序是差不多的, 都是有4步, 其中有两步它们两个是一样的, 另外两步虽然具体内容不一样, 但是都做做的同一类工作.
1382 0
|
C#
使用C# (.NET Core) 实现命令设计模式 (Command Pattern)
本文的概念内容来自深入浅出设计模式一书. 项目需求 有这样一个可编程的新型遥控器, 它有7个可编程插槽, 每个插槽可连接不同的家用电器设备. 每个插槽对应两个按钮: 开, 关(ON, OFF).
860 0
|
Java C#
使用C# (.NET Core) 实现单体设计模式 (Singleton Pattern)
本文的概念内容来自深入浅出设计模式一书 由于我在给公司做内培, 所以最近天天写设计模式的文章.... 单体模式 Singleton 单体模式的目标就是只创建一个实例. 实际中有很多种对象我们可能只需要它们的一个实例, 例如: 线程池,缓存, 弹出的对话框, 用于保存设置的类, 用于logging的类, 硬件设备驱动对象等等.
1228 0
|
C#
使用C# (.NET Core) 实现抽象工厂设计模式 (Abstract Pattern)
本文的概念性内容来自深入浅出设计模式一书. 上一篇文章讲了简单工厂和工厂方法设计模式 http://www.cnblogs.com/cgzl/p/8760250.html, 使用的是披萨店的例子. 文将继续使用这个例子, 这里要用到抽象工厂.
1373 0
|
C# 设计模式 .NET
使用C# (.NET Core) 实现简单工厂(Simple Factory) 和工厂方法设计模式 (Factory Method Pattern)
本文源自深入浅出设计模式. 只不过我是使用C#/.NET Core实现的例子.   前言 当你看见new这个关键字的时候, 就应该想到它是具体的实现. 这就是一个具体的类, 为了更灵活, 我们应该使用的是接口(interface).
1460 0
|
安全 C# 数据安全/隐私保护
使用C# (.NET Core) 实现装饰模式 (Decorator Pattern) 并介绍 .NET/Core的Stream
该文章综合了几本书的内容. 某咖啡店项目的解决方案 某咖啡店供应咖啡, 客户买咖啡的时候可以添加若干调味料, 最后要求算出总价钱. Beverage是所有咖啡饮料的抽象类, 里面的cost方法是抽象的.
1393 0