草庐IT

C# 设计模式——生成器模式(建造者模式)、适配器模式

魏杨杨 2023-03-28 原文

1、生成器模式

      我们知道工厂模式能够根据传递给构造方法返回几个不同子类中的一个,假设我们不但需要一个用于计算的算法,还要根据显示的数据不通而现实不同的界面,我们就知道改如何使用生成器模式(Builder Pattern)。生成器模式将一个复杂对象的构建与它的表示分离,使得同样的构建过程可以创建不同的表示我们用个栗子看看。还是造车。

/// <summary>
/// 造车步骤
/// </summary>
public abstract class Builder
{
    /// <summary>
    /// 造轮子
    /// </summary>
    public abstract void MakeWheels();
    /// <summary>
    /// 组装
    /// </summary>
    public abstract void Assemble();
    /// <summary>
    /// 喷漆
    /// </summary>
    public abstract void Painting();
}

     再写一个造大车跟造小车的类

/// <summary>
/// 造大车
/// </summary>
public class BigBuilder : Builder
{
    public override void Assemble()
    {
        Console.WriteLine("大车组装完成啦");
    }

    public override void MakeWheels()
    {
        Console.WriteLine("大车轮子造好啦!");
    }

    public override void Painting()
    {
        Console.WriteLine("大车喷漆完成!");
    }
}
/// <summary>
/// 造小车
/// </summary>
public class SmallBuilder : Builder
{
    public override void Assemble()
    {
        Console.WriteLine("小车组装完成啦");
    }

    public override void MakeWheels()
    {
        Console.WriteLine("小车轮子造好啦!");
    }

    public override void Painting()
    {
        Console.WriteLine("小车喷漆完成!");
    }
}

     后面加一个工厂 造车的。它使用Builder来创建不同种类的汽车

/// <summary>
/// 造车工厂
/// </summary>
public class CarFactory
{
    public Builder Builder { get; set; }
    public CarFactory(Builder builder)
    {
        Builder=builder;
    }
    /// <summary>
    /// 造车
    /// </summary>
    public void CreateCar()
    {
        Builder.MakeWheels();//轮子
        Builder.Assemble();//组装
        Builder.Painting();//喷漆
    }
    

}

//客户端调用
CarFactory cf = new(new BigBuilder());
cf.CreateCar();
cf.Builder = new SmallBuilder();
cf.CreateCar();

     所以可以看出生成器模式的好处:

    允许读者改变产品内部表示,同事也影藏了产品如何装的问题;

    每个特定的生成器都独立于其他的生成器,同事独立于程序的其他部分。所以能精确的控制生成器构建的结果;

    生成器的模式类似于抽象工厂模式,两者都返回了相同的方法个对象组成的类。他们之间主要的差别是抽象工厂返回一一系列相关的类而生成器是根据它提供的数据一步一步的构建出一个复杂的对象。 

  建造者模式的缺点:

    当组件的组合种类很多时,需要创建很多的具体建造者类。

2、适配器模式

       适配器模式(Adapter  Pattern)可以将一个程序设计接口转换为另外一个接口。当我们想让不相关的类在同一个程序里面一起工作的时候就可以用适配器的模式。一是继承可以实现,从一个不一致的类派生出一个新类添加需要的方法让派生类满足所需要的接口;二是对象组合,就是把原始类包含在新类里面,然后再新类里面创建方法转换调用。这两种方法统称为类适配器和对象适配器。说着有点指鹿为马的感觉了(程序中就是这样的)。这里有三个角色。

   初始角色:我们需要的功能,但是接口不匹配;

   目标角色:他就表现了客户需要的接口

   适配器觉得:实现了目标接口

   举个栗子 我是苹果手机想充钱,但是我只有安卓的充电线,这时候是不是就要找一个转接头来我们也可以充电了。

/// <summary>
/// 安卓充电
/// </summary>
public class Android
{
    public void AndroidCharge()
    {
        Console.WriteLine("安卓充电");
    }
}
/// <summary>
/// 苹果接口充电
/// </summary>
public interface IApple
{
    /// <summary>
    /// 目标方法
    /// </summary>
    void AppleChage();
}
public class AppleAdater : IApple
{
    Android ad = new Android();
    /// <summary>
    /// 评估充电,实际上是调用安卓的充电方法
    /// </summary>
    public void AppleChage()
    {
        ad.AndroidCharge();
    }
}

//客户端调用
IApple app = new AppleAdater();
app.AppleChage();

适配器模式的优点:

  1.提高代码复用,复用了Adaptee中的方法

  2.灵活性好,如没有关联的两个类A和类B,类A想借用类B的方法,可以在类A中添加一个类B的实例,然后调用类B实例的方法即可。

适配器模式的缺点:

  1.过多使用适配器会让会同凌乱,如我们明明调用的是A接口的功能,但是却被适配成了B接口的功能。

 

PS:每个人都应该把自己的每一天当成一场自我的创作,在这创作中诚实的体现自己,而且把这一天当作是一个无法重复的一天,抱持着不能复制的心情过好这一天。

有关C# 设计模式——生成器模式(建造者模式)、适配器模式的更多相关文章

  1. ruby-on-rails - Rails - 子类化模型的设计模式是什么? - 2

    我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co

  2. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  3. ruby - capybara field.has_css?匹配器 - 2

    我在MiniTest::Spec和Capybara中使用以下规范:find_field('Email').must_have_css('[autofocus]')检查名为“电子邮件”的字段是否具有autofocus属性。doc说如下:has_css?(path,options={})ChecksifagivenCSSselectorisonthepageorcurrentnode.据我了解,字段“Email”是一个节点,因此调用must_have_css绝对有效!我做错了什么? 最佳答案 通过JonasNicklas得到了答案:No

  4. ruby - 如何在续集中重新加载表模式? - 2

    鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende

  5. ruby-on-rails - 使用 rails 4 设计而不更新用户 - 2

    我将应用程序升级到Rails4,一切正常。我可以登录并转到我的编辑页面。也更新了观点。使用标准View时,用户会更新。但是当我添加例如字段:name时,它​​不会在表单中更新。使用devise3.1.1和gem'protected_attributes'我需要在设备或数据库上运行某种更新命令吗?我也搜索过这个地方,找到了许多不同的解决方案,但没有一个会更新我的用户字段。我没有添加任何自定义字段。 最佳答案 如果您想允许额外的参数,您可以在ApplicationController中使用beforefilter,因为Rails4将参数

  6. c# - 如何在 ruby​​ 中调用 C# dll? - 2

    如何在ruby​​中调用C#dll? 最佳答案 我能想到几种可能性:为您的DLL编写(或找人编写)一个COM包装器,如果它还没有,则使用Ruby的WIN32OLE库来调用它;看看RubyCLR,其中一位作者是JohnLam,他继续在Microsoft从事IronRuby方面的工作。(估计不会再维护了,可能不支持.Net2.0以上的版本);正如其他地方已经提到的,看看使用IronRuby,如果这是您的技术选择。有一个主题是here.请注意,最后一篇文章实际上来自JohnLam(看起来像是2009年3月),他似乎很自在地断言RubyCL

  7. C# 到 Ruby sha1 base64 编码 - 2

    我正在尝试在Ruby中复制Convert.ToBase64String()行为。这是我的C#代码:varsha1=newSHA1CryptoServiceProvider();varpasswordBytes=Encoding.UTF8.GetBytes("password");varpasswordHash=sha1.ComputeHash(passwordBytes);returnConvert.ToBase64String(passwordHash);//returns"W6ph5Mm5Pz8GgiULbPgzG37mj9g="当我在Ruby中尝试同样的事情时,我得到了相同sha

  8. ruby - 是否有用于序列化和反序列化各种格式的对象层次结构的模式? - 2

    给定一个复杂的对象层次结构,幸运的是它不包含循环引用,我如何实现支持各种格式的序列化?我不是来讨论实际实现的。相反,我正在寻找可能会派上用场的设计模式提示。更准确地说:我正在使用Ruby,我想解析XML和JSON数据以构建复杂的对象层次结构。此外,应该可以将该层次结构序列化为JSON、XML和可能的HTML。我可以为此使用Builder模式吗?在任何提到的情况下,我都有某种结构化数据-无论是在内存中还是文本中-我想用它来构建其他东西。我认为将序列化逻辑与实际业务逻辑分开会很好,这样我以后就可以轻松支持多种XML格式。 最佳答案 我最

  9. ruby-on-rails - 如何在 Rails 3 中创建自定义脚手架生成器? - 2

    有这些railscast。http://railscasts.com/episodes/218-making-generators-in-rails-3有了这个,你就会知道如何创建样式表和脚手架生成器。http://railscasts.com/episodes/216-generators-in-rails-3通过这个,您可以了解如何添加一些文件来修改脚手架View。我想把两者结合起来。我想创建一个生成器,它也可以创建脚手架View。有点像RyanBates漂亮的生成器或web_app_themegem(https://github.com/pilu/web-app-theme)。我

  10. 基于C#实现简易绘图工具【100010177】 - 2

    C#实现简易绘图工具一.引言实验目的:通过制作窗体应用程序(C#画图软件),熟悉基本的窗体设计过程以及控件设计,事件处理等,熟悉使用C#的winform窗体进行绘图的基本步骤,对于面向对象编程有更加深刻的体会.Tutorial任务设计一个具有基本功能的画图软件**·包括简单的新建文件,保存,重新绘图等功能**·实现一些基本图形的绘制,包括铅笔和基本形状等,学习橡皮工具的创建**·设计一个合理舒适的UI界面**注明:你可能需要先了解一些关于winform窗体应用程序绘图的基本知识,以及关于GDI+类和结构的知识二.实验环境Windows系统下的visualstudio2017C#窗体应用程序三.

随机推荐