[Padrão de design criativo] O padrão de fábrica do padrão de design C# e a realização de fábrica dinâmica por meio da reflexão.

Os tópicos são os seguintes:

假设你正在为一家汽车制造公司编写软件。公司生产多种类型的汽车,包括轿车、SUV和卡车。每种汽车都有不同的特点和功能。

请设计一个工厂模式,用于创建不同类型的汽车对象。该工厂模式应具有以下要求:

工厂类名为 CarFactory,包含一个静态方法 CreateCar,根据传入的参数类型,返回对应类型的汽车对象。
汽车类 Car 是一个抽象类,包含一个抽象方法 Drive,用于描述汽车的驾驶行为。
轿车类 Sedan 继承自 Car 类,实现了 Drive 方法,在 Drive 方法中输出“驾驶轿车”。
SUV 类继承自 Car 类,实现了 Drive 方法,在 Drive 方法中输出“驾驶SUV”。
卡车类 Truck 继承自 Car 类,实现了 Drive 方法,在 Drive 方法中输出“驾驶卡车”。
使用该工厂模式完成以下操作:

在程序入口处,向 CarFactory 的 CreateCar 方法传入参数 "sedan",并将返回的对象存储到 sedanCar 变量中。
在 sedanCar 上调用 Drive 方法,观察输出结果。
同样地,创建一个 SUV 对象,并调用其 Drive 方法。
创建一个 Truck 对象,并调用其 Drive 方法。
请根据以上要求实现该工厂模式,并编写相应的代码。

Implementação simples de fábrica:

using System;
using System.Reflection;


namespace Factory
{
    
    
    internal class CarFactory
    {
    
    
        public static T CreateCar<T>() where T : Car, new()
        {
    
    
            if (typeof(T) == typeof(Sedan))
                return new Sedan() as T;
            if (typeof(T) == typeof(SUV))
                return new SUV() as T;
            if (typeof(T) == typeof(Truck))
                return new Truck() as T;

            throw new ArgumentException("无法创建该类型的汽车对象。");
        }

     

    }
}

namespace Factory
{
    
    
    public abstract class Car 
    {
    
    

        public abstract void Drive();      
    }
}

using System;


namespace Factory
{
    
    
    internal class Program
    {
    
    
        static void Main(string[] args)
        {
    
    
            Car sedanCar = CarFactory.CreateCar<Sedan>();
            sedanCar.Drive();

            Car suvCar = CarFactory.CreateCar<SUV>();
            suvCar.Drive();

            Car truckCar = CarFactory.CreateCar<Truck>();
            truckCar.Drive();




            Console.ReadLine();
        }
    }
}

O código acima fornece a definição básica da classe base abstrata, a implementação da fábrica genérica e o método de chamada.
Vale a pena notar que where T : Car, new()esta condição
where T : Car, new()é uma restrição aos parâmetros de tipo genérico T. Esta restrição afirma que os parâmetros de tipo genérico Tdevem satisfazer duas condições:

TDeve ser Caruma classe ou sua classe derivada: significa que Tdeve ser Carum tipo ou Caruma subclasse (classe derivada) da classe.

TDeve ter um construtor sem argumento: Através new()da restrição, significa que Tdeve ter um construtor sem argumento, ou seja, ser capaz de usar new T() 创建 Tuma instância de .

O objetivo das restrições acima é garantir que os parâmetros de tipo genérico Tatendam aos requisitos durante o uso. A restrição Tdeve ser Cardo tipo ou de seu tipo derivado para garantir que CreateCaro objeto criado no método seja Caruma instância do tipo ou de sua subclasse; ao mesmo tempo, a restrição Tdeve ter um construtor sem argumentos para garantir que uma instância de possa ser new T()criado por T.

Por meio de tais restrições, pode-se garantir que, ao chamar CreateCaro método, os parâmetros de tipo passados T​​​​atendam aos requisitos especificados e Ta instância do objeto correspondente possa ser criada com base no tipo dentro do método.

工厂模式适用于以下场景:

对象创建复杂:当对象的创建涉及到复杂的逻辑判断、依赖关系或者大量的初始化操作时,可以使用工厂模式来封装对象的创建过程。这样可以简化客户端代码,并提供一个统一的入口来创建对象。

对象类型不确定:当需要根据不同的条件或者配置来创建不同类型的对象时,可以使用工厂模式。通过工厂模式,可以将对象的创建逻辑从客户端代码中分离出来,降低代码的耦合性。

扩展性要求高:当系统中需要添加新的产品或者变化频繁时,使用工厂模式可以方便地扩展和修改代码。通过增加新的具体工厂和产品类,而无需修改已有代码,符合开闭原则。

隐藏对象创建细节:当需要隐藏对象创建的细节时,可以使用工厂模式。客户端只需要与工厂接口进行交互,无需关心具体的产品如何创建或实现。

总的来说,工厂模式适用于对象创建复杂、对象类型不确定、扩展性要求高以及隐藏对象创建细节的情况。它能够提供灵活、可扩展和易于维护的代码结构。

A partir do código, podemos ver que CarFactoryesta classe de fábrica tem desvantagens: por exemplo, cada vez que adicionamos um novo objeto carro, precisamos modificar CarFactorya classe CreateCare adicionar um novo objeto. Neste momento podemos usar uma fábrica dinâmica para criar objetos através da reflexão.

using System;
using System.Reflection;


namespace Factory
{
    
    
    internal class CarFactory
    {
    
    

        public static T CreateCar<T>(string typeName) where T : Car, new()
        {
    
    
            Type type = GetTypeByName(typeName);
            if (type != null && typeof(T).IsAssignableFrom(type))  return Activator.CreateInstance(type) as T;

            throw new ArgumentException("无法创建该类型的汽车对象。");
        }

        private static Type GetTypeByName(string typeName)
        {
    
    
            return Assembly.GetAssembly(typeof(CarFactory)).GetType($"Factory.{
      
      typeName}");
        }

    }
}

using System;


namespace Factory
{
    
    
    internal class Program
    {
    
    
        static void Main(string[] args)
        {
    
    

            // 传递具体类的类型名称作为参数
            string typeName = "Sedan";
            Car sedan = CarFactory.CreateCar<Sedan>(typeName);
            sedan.Drive();
            // 可以根据需要创建不同类型的汽车对象
            typeName = "SUV";
            Car suv = CarFactory.CreateCar<SUV>(typeName);
            suv.Drive();
            typeName = "Truck";
            Car truck = CarFactory.CreateCar<Truck>(typeName);
            truck.Drive();

            Console.ReadLine();
        }
    }
}

Dessa forma, não precisamos modificar o código principal da fábrica. Gere dinamicamente uma instância
passando o assembly onde o tipo atual está localizado. Também significa se o tipo atual pode ser convertido em outro tipo, como se é uma classe base ou uma classe derivada de T. Usado para criar instâncias.typeNameAssembly.GetAssembly(typeof(CarFactory))Factory.{typeName} 为 命名空间.类typeof(T).IsAssignableFrom(type)TTypetypeActivator.CreateInstance(type) as T

Acho que você gosta

Origin blog.csdn.net/csdn2990/article/details/132082222
Recomendado
Clasificación