Unity 设计模式 之 创建型模式 -【单例模式】【原型模式】 【建造者模式】

news2024/9/20 0:54:39

Unity 设计模式 之 创建型模式 -【单例模式】【原型模式】 【建造者模式】

目录

Unity 设计模式 之 创建型模式 -【单例模式】【原型模式】 【建造者模式】

一、简单介绍

二、单例模式 (Singleton Pattern)

1、什么时候使用单例模式

2、单例模式的好处

3、使用单例模式的注意事项

三、在 Unity 中使用 单例模式

1、普通型(new)泛型单例模式

2、继承 MonoBehaviour 的泛型单例模式

3、单例使用方式

4、实现分析

四、原型模式 (Prototype Pattern)

1、原型模式的原理

2、什么时候使用原型模式

3、使用原型模式的好处

4、使用原型模式的注意事项

五、在 Unity 中使用 原型模式

1、 定义基类 ShapePrototype

2、创建具体的形状类

3、创建 ShapeSpawner 负责克隆对象

4、设置场景中的原型对象

5、运行效果

六、建造者模式 (Builder Pattern)

1、什么时候使用建造者模式

2、使用建造者模式的好处

3、建造者模式的注意事项

七、在 Unity 中使用 建造者模式

1、定义房屋类 House

2、定义建造者接口 IHouseBuilder

3、 具体建造者类:木屋建造者 WoodenHouseBuilder 和砖屋建造者 BrickHouseBuilder

4、定义指挥者 Director

5、 在 Unity 场景中使用建造者模式

6、运行效果


一、简单介绍

设计模式 是指在软件开发中为解决常见问题而总结出的一套 可复用的解决方案。这些模式是经过长期实践证明有效的 编程经验总结,并可以在不同的项目中复用。设计模式并不是代码片段,而是对常见问题的 抽象解决方案,它提供了代码结构和模块间交互的一种设计思路,帮助开发者解决特定的设计问题。

设计模式的特点:

  1. 通用性:设计模式针对的是软件开发中常见的设计问题,适用于各种软件工程项目。
  2. 可复用性:设计模式可以在不同项目和环境下被重复使用,提高代码的可维护性和扩展性。
  3. 可扩展性:设计模式有助于让代码结构更加灵活,易于扩展和修改。
  4. 模块化:通过设计模式,可以减少代码的耦合性,增强模块间的独立性。
  5. 提高沟通效率:设计模式为开发者提供了一种通用的设计语言,使得团队成员能够快速理解并讨论设计方案。

二、单例模式 (Singleton Pattern)

单例模式 (Singleton Pattern) 是一种创建型设计模式,保证一个类只有一个实例,并提供一个全局访问点来获取该实例。它通过控制类的实例化过程,确保系统中只有一个该类的对象存在。

在单例模式中,类的构造函数通常是私有的,防止外部通过 new 来创建对象,类内部维护一个静态实例,通过公共的静态方法提供访问。

单例模式的特点

  1. 唯一实例:单例模式确保一个类只有一个实例,无论系统中的组件如何调用它。
  2. 全局访问点:单例模式为客户端提供一个全局访问点(通常是静态方法),通过这个访问点,所有客户端都能获得相同的实例。
  3. 延迟初始化:可以在首次访问时创建该实例,节省系统资源(可选)。

1、什么时候使用单例模式

  1. 全局唯一的资源管理:如果一个类需要控制对某种共享资源(如数据库连接池、文件系统、日志器、配置管理器等)的访问,并且不允许有多个实例同时存在。

  2. 配置或状态共享:当系统中某个类需要持有全局配置或共享状态时,单例模式可以使得这些状态在不同模块中保持一致。

  3. 资源开销较大的对象:如果一个类的实例化代价较高(如外部资源初始化、复杂运算等),而且只需要一个实例时,单例模式可以避免多次重复的创建操作。

  4. 需要严格控制实例数量:在某些业务逻辑中,严格要求一个类只能有一个实例,否则会引发逻辑混乱或资源竞争(如操作系统中的任务管理器)。

2、单例模式的好处

  1. 控制实例数量:确保一个类只生成一个实例,避免不必要的内存占用,尤其对于管理全局状态或资源的对象(如数据库连接、日志系统、配置管理等),控制实例数量可以提高系统的稳定性。

  2. 全局访问:单例模式提供了一个全局访问点,这使得某些系统级别的服务(如日志系统、资源管理器等)能够方便地被全局访问。

  3. 节省资源:避免重复创建相同对象,减少了资源的消耗,尤其是对于耗费较大的对象(如文件系统、数据库连接等)。

  4. 线程安全:通过实现线程安全的单例,可以确保在多线程环境下,多个线程对该对象的操作是安全的。

3、使用单例模式的注意事项

  1. 全局状态的复杂性:虽然单例模式提供全局访问,但这也意味着它引入了全局状态,可能导致系统的状态管理变得复杂,特别是在大型系统中。

  2. 并发和线程安全:在多线程环境下,需要确保单例的实现是线程安全的,否则可能会出现多个线程同时创建实例的情况,导致数据不一致。

  3. 难以扩展:由于单例模式限制了对象的创建数量,在某些情况下,可能不利于系统的扩展或测试。比如,在单元测试中,单例模式可能难以进行模拟。

  4. 滥用风险:单例模式提供全局访问点,容易被滥用为全局变量,违背面向对象设计中高内聚、低耦合的原则。因此,使用时应考虑是否真的有必要将类设计为单例。

总之,单例模式 主要用于控制对象的实例化,确保系统中只有一个类的实例,并通过全局访问点来控制对象的使用。它适用于需要全局共享资源、统一管理的场景,如日志系统、数据库连接池等。尽管单例模式在某些场景下有助于提升系统的稳定性和效率,但也应谨慎使用,以避免全局状态管理复杂化或滥用全局访问带来的耦合问题。

三、在 Unity 中使用 单例模式

在 Unity 中,实现一个线程安全的普通类MonoBehaviour 类的泛型单例时,必须考虑以下几点:

  1. 普通类单例:不能被 new,并且在多线程环境下线程安全。
  2. MonoBehaviour 单例:由于 MonoBehaviour 的实例是通过 Unity 的 AddComponent 创建的,不能直接通过 new,也需要保证在多线程环境下的安全性。

1、普通型(new)泛型单例模式

方法一:
public abstract class Singleton<T> where T : class, new()
{
    private static T instance = null;
 
    // 多线程安全机制
    private static readonly object locker = new object();
 
    public static T Instance
    {
        get
        {
            lock (locker)
            {
                if (instance == null)
                    instance = new T();
                return instance;
            }
        }
    }
}
 
 
 
方法二:
 
using System;
using System.Reflection;
 
/// <summary>
/// 单例
/// 继承需要一个非公有的无参构造函数
/// </summary>
/// <typeparam name="T">类名的泛型</typeparam>
public abstract class Singleton<T> where T : class
{
    private static T instance = null;
 
    // 多线程安全机制
    private static readonly object locker = new object();
 
    public static T Instance
    {
        get
        {
            // 线程锁
            lock (locker)
            {
                if (null == instance)
                {
                    // 反射获取实例
                    var octors = typeof(T).GetConstructors(BindingFlags.Instance | BindingFlags.NonPublic) ;
 
                    // 获取无参数的非公共构造函数
                    var octor = Array.Find(octors, c => c.GetParameters().Length == 0);
 
                    // 没有则提示没有私有的构造函数
                    if (null == octor)
                    {
                        throw new Exception("No NonPublic constructor without 0 parameter");
                    }
 
                    // 实例化
                    instance = octor.Invoke(null) as T;
                }
 
                return instance;
 
            }
        }
    }
 
    /// <summary>
    /// 构造函数
    /// 避免外界new
    /// </summary>
    protected Singleton() { }
}

2、继承 MonoBehaviour 的泛型单例模式

using UnityEngine;
 
public abstract class MonoSingleton<T> : MonoBehaviour where T : MonoBehaviour
{
    private static T instance = null;
 
    private static readonly object locker = new object();
 
    private static bool bAppQuitting;
 
    public static T Instance
    {
        get
        {
            if (bAppQuitting)
            {
                instance = null;
                return instance;
            }
 
            lock (locker)
            {
                if (instance == null)
                {
                    // 保证场景中只有一个 单例
                    T[] managers = Object.FindObjectsOfType(typeof(T)) as T[];
                    if (managers.Length != 0)
                    {
                        if (managers.Length == 1)
                        {
                            instance = managers[0];
                            instance.gameObject.name = typeof(T).Name;
                            return instance;
                        }
                        else
                        {
                            Debug.LogError("Class " + typeof(T).Name + " exists multiple times in violation of singleton pattern. Destroying all copies");
                            foreach (T manager in managers)
                            {
                                Destroy(manager.gameObject);
                            }
                        }
                    }
 
 
                    var singleton = new GameObject();
                    instance = singleton.AddComponent<T>();
                    singleton.name = "(singleton)" + typeof(T);
                    singleton.hideFlags = HideFlags.None;
                    DontDestroyOnLoad(singleton);
 
                }
                instance.hideFlags = HideFlags.None;
                return instance;
            }
        }
    }
 
    protected virtual void Awake()
    {
        bAppQuitting = false;
    }
 
    protected virtual void OnDestroy()
    {
        bAppQuitting = true;
    }
}

3、单例使用方式

继承泛型单例模式即可

 
public class Test1 : MonoSingleton<Test1>{}
 
// 方法一
public class Test2 : Singleton<Test2>{}
// 方法二
public class Test2 : Singleton<Test2>{{private Test2(){}}

4、实现分析

  1. 普通类单例

    • 通过 lock 实现线程安全,避免了多线程环境下重复创建实例的问题。
    • 将构造函数设为 protected,确保外部不能通过 new 关键字实例化该类。
    • 在 Unity 中,适合管理游戏状态、配置数据等不需要与 Unity 引擎生命周期强绑定的对象。
  2. MonoBehaviour 类单例

    • 使用了 lock 机制确保多线程环境中的安全性。
    • 使用 FindObjectOfType<T> 检查场景中是否已经有该类型的实例,如果没有则创建新的对象并添加该组件。
    • 通过 DontDestroyOnLoad 确保实例在场景切换时不被销毁,适合音频管理器、游戏控制器等需要跨场景保持的对象。

这两种单例的模式在 Unity 中广泛适用,可以有效地管理全局对象和跨场景对象。

四、原型模式 (Prototype Pattern)

原型模式 (Prototype Pattern) 是一种创建型设计模式,它允许通过复制现有的对象来创建新对象,而不是通过直接实例化类。这意味着你可以通过克隆原型对象来生成新的实例,而不必依赖类的构造函数。该模式的核心思想是,通过创建一个对象的副本(即原型)来避免昂贵的初始化操作。

在 C# 中,通常通过实现 ICloneable 接口或者自定义的克隆方法来实现原型模式。

1、原型模式的原理

原型模式的工作原理是:

  1. 定义一个原型接口或基类,用来提供克隆的方法(如 Clone)。
  2. 具体的类实现该接口,提供自己的克隆方法,该方法返回当前对象的深拷贝或浅拷贝。
  3. 客户端可以通过克隆原型对象来创建新对象,而无需知道如何构造对象。

拷贝的类型可以分为:

  • 浅拷贝 (Shallow Copy):只复制对象的基本数据类型,对于引用类型的字段,只复制引用地址。
  • 深拷贝 (Deep Copy):不仅复制对象的所有值,还递归地复制所有引用对象,创建独立的副本。

2、什么时候使用原型模式

  1. 需要频繁创建相似对象时:如果系统需要创建大量相似或结构复杂的对象时,通过克隆原型对象来生成新对象可以提高性能。

  2. 对象的创建代价高昂时:如果对象的初始化非常复杂,包含很多属性设置或涉及昂贵的资源操作(如网络连接、文件系统操作),使用原型模式避免了重复创建过程。

  3. 避免对象过多的子类化:当不希望为了创建新对象而引入更多的子类或扩展类时,原型模式可以避免通过继承创建不同类型对象的繁琐过程。

  4. 需要保存对象的历史状态或备份时:如果需要将某个对象的状态备份或在某个时刻进行复制(如撤销操作、快照功能),可以通过原型模式来克隆对象。

3、使用原型模式的好处

  1. 提高对象创建效率:对于某些对象的创建过程非常复杂或耗时时,通过克隆现有对象(即原型)可以避免重复的复杂初始化,节省时间和资源。

  2. 避免子类膨胀:通过原型模式,程序不需要通过继承或工厂模式来生成对象,这可以减少子类的数量,避免继承层次的膨胀。

  3. 动态生成对象:原型模式允许在运行时动态生成对象,不需要事先知道具体类的名称。

  4. 减少对象依赖:克隆对象是独立于具体类的,原型模式不需要直接依赖构造函数,这降低了对象之间的耦合度。

  5. 灵活性高:通过克隆,可以对现有的对象进行修改后创建新的对象,特别适合那些需要频繁修改对象属性的场景。

4、使用原型模式的注意事项

  1. 克隆的深浅拷贝:对于引用类型字段,使用浅拷贝时需要特别小心,浅拷贝只复制引用,而不复制对象本身,可能导致两个对象共用同一份数据。若需要完全独立的对象,必须使用深拷贝。

  2. 复杂对象的克隆:当对象包含复杂的嵌套结构或其他依赖时,确保实现合适的深拷贝逻辑,否则可能会出现数据共享或数据覆盖问题。

  3. 性能考虑:尽管原型模式避免了对象的重复构造,但深拷贝可能引入较大的性能开销,特别是在对象嵌套复杂时。因此,在性能敏感的场景下要慎重考虑深拷贝的实现。

  4. 原型的可变性:当通过原型模式创建对象时,如果不慎修改了原型本身的状态,所有基于该原型创建的对象也可能受到影响。因此在设计时,需要确保原型对象的状态是安全的。

总之,原型模式 通过复制现有对象来生成新对象,避免了类构造的开销,特别适用于对象创建代价高昂或需要动态创建对象的场景。它提供了灵活的对象创建方式,减少了类的复杂性和耦合度。使用时需要根据具体需求选择浅拷贝或深拷贝,并确保对象的可复制性和独立性。

五、在 Unity 中使用 原型模式

在 Unity 中,原型模式可以应用于场景中需要频繁生成的对象,比如 3D 模型(如立方体、球体等)。通过原型模式,我们可以避免每次都从头实例化对象,而是通过复制现有的对象来创建新的实例。

我们将创建一个简单的原型模式应用,用于克隆 Unity 中的 3D 对象(立方体、球体等),并根据用户输入生成多个克隆对象。

步骤:

  1. 创建一个基类 ShapePrototype,它提供克隆接口。
  2. 创建不同的形状类,如立方体和球体,继承自 ShapePrototype
  3. 使用原型模式通过克隆现有对象来创建新对象,而不是直接创建新对象。

1、 定义基类 ShapePrototype

using UnityEngine;

// 定义一个抽象的原型基类
public abstract class ShapePrototype : MonoBehaviour
{
    // 定义一个抽象的克隆方法
    public abstract ShapePrototype Clone();

    // 通用的显示形状信息的方法
    public abstract void DisplayInfo();
}

2、创建具体的形状类

2.1 立方体类

using UnityEngine;

public class Cube : ShapePrototype
{
    // 重写克隆方法
    public override ShapePrototype Clone()
    {
        // 实现浅拷贝,复制当前对象的属性
        GameObject clone = Instantiate(this.gameObject);
        return clone.GetComponent<ShapePrototype>();
    }

    public override void DisplayInfo()
    {
        Debug.Log("This is a Cube.");
    }
}

2.2 球体类

using UnityEngine;

public class Sphere : ShapePrototype
{
    // 重写克隆方法
    public override ShapePrototype Clone()
    {
        // 实现浅拷贝,复制当前对象的属性
        GameObject clone = Instantiate(this.gameObject);
        return clone.GetComponent<ShapePrototype>();
    }

    public override void DisplayInfo()
    {
        Debug.Log("This is a Sphere.");
    }
}

3、创建 ShapeSpawner 负责克隆对象

using UnityEngine;

public class ShapeSpawner : MonoBehaviour
{
    public ShapePrototype cubePrototype;   // 立方体原型
    public ShapePrototype spherePrototype; // 球体原型

    private void Update()
    {
        // 按下 C 键克隆立方体
        if (Input.GetKeyDown(KeyCode.C))
        {
            ShapePrototype cubeClone = cubePrototype.Clone();
            cubeClone.transform.position = new Vector3(Random.Range(-5, 5), 1, Random.Range(-5, 5));
            cubeClone.DisplayInfo();
        }

        // 按下 S 键克隆球体
        if (Input.GetKeyDown(KeyCode.S))
        {
            ShapePrototype sphereClone = spherePrototype.Clone();
            sphereClone.transform.position = new Vector3(Random.Range(-5, 5), 1, Random.Range(-5, 5));
            sphereClone.DisplayInfo();
        }
    }
}

4、设置场景中的原型对象

  1. 创建一个 Unity 场景,并添加一个空物体 ShapeSpawner,将上面的 ShapeSpawner 脚本挂载到该物体上。
  2. 在场景中创建一个立方体和一个球体,并将它们的 CubeSphere 脚本分别挂载到立方体和球体上。
  3. ShapeSpawner 脚本的 cubePrototypespherePrototype 变量中,分别拖拽场景中的立方体和球体作为原型对象。

5、运行效果

  • 按下 C 键:克隆一个新的立方体,随机放置在场景中,并在控制台输出 "This is a Cube."
  • 按下 S 键:克隆一个新的球体,随机放置在场景中,并在控制台输出 "This is a Sphere."

每次克隆都是基于场景中的原型对象,这样可以避免重新生成对象的开销,并且保证克隆对象与原型对象的所有属性相同。

节省资源,通过克隆现有对象,而不是每次都从头实例化对象,可以提高性能,尤其是在场景中需要大量生成对象时;灵活性,可以动态调整原型对象的属性,并通过克隆生成新的实例,而不需要修改对象的创建逻辑;方便扩展,如果需要新的形状,可以轻松扩展原型类,无需修改现有代码。

这种模式在 Unity 的游戏开发中非常适合用于生成大量相似对象(如敌人、物品、特效等)。

六、建造者模式 (Builder Pattern)

建造者模式 (Builder Pattern) 是一种创建型设计模式,它将复杂对象的构建过程与对象的表示分离,使得同样的构建过程可以创建不同的对象。建造者模式特别适用于那些构建步骤复杂且具有多种配置的对象。

在建造者模式中,通常有以下几个关键部分:

  1. Builder (建造者):定义了创建产品对象的步骤,通常是一个抽象类或接口。
  2. Concrete Builder (具体建造者):实现了 Builder 接口的类,负责具体的构建细节,创建具体的产品对象。
  3. Director (指挥者):负责控制构建过程,按照一定的顺序调用 Builder 的方法来构建对象。
  4. Product (产品对象):最终构建的复杂对象。

1、什么时候使用建造者模式

  1. 构建复杂对象时:如果一个对象的创建步骤非常复杂,需要按照一定的顺序构建多个部件,建造者模式可以帮助你将这些步骤组织清晰,避免直接调用复杂的构造函数。

  2. 需要生成多种类型对象时:当需要通过相同的构建过程生成不同类型或配置的对象时,建造者模式可以帮助你在同一个框架下灵活构建不同的产品。

  3. 构造函数参数过多时:如果一个类的构造函数有很多可选参数或者参数组合非常复杂,使用建造者模式可以避免构造函数的复杂性,通过链式调用来简化参数设置。

  4. 对象的创建流程固定但表示方式多样时:在一些场景下,产品的创建过程是相同的,但是生成的产品可能有不同的表现方式或配置。此时,建造者模式可以将构建流程抽象化,从而生成不同的产品。

2、使用建造者模式的好处

  1. 简化对象构建过程:建造者模式将复杂对象的构建步骤集中在一个地方,减少了创建对象的复杂度,尤其是当对象拥有多个可选参数或需要按特定步骤构建时。

  2. 灵活构建复杂对象:通过建造者模式,可以将对象的构建过程拆分成多个步骤,允许构建者在不同的构建过程中创建不同类型的对象或者配置不同的参数。

  3. 代码清晰、易于维护:建造者模式使代码更清晰,因为对象的构建逻辑被独立封装在 Builder 中,而不是混杂在产品对象内部。这使得代码易于维护和扩展。

  4. 支持不同的产品变化:同样的构建过程可以生成不同的产品,通过使用不同的建造者,能够灵活应对产品的变化需求。

  5. 简化对象的可变参数管理:如果对象有很多可选参数,使用建造者模式可以避免复杂的构造函数,提供链式调用来设置参数,使得代码更加简洁。

3、建造者模式的注意事项

  1. 多个建造者的协作:如果你需要构建多个复杂对象,可以为每种对象提供不同的具体建造者,并通过同一个指挥者来协调构建过程。

  2. 对象的不可变性:建造者模式可以配合不可变对象使用,确保对象在构建完成后无法被修改。

  3. 建造顺序的灵活性:确保建造者的接口设计支持灵活的建造顺序,允许客户端根据需要选择不同的建造步骤。

总之,建造者模式通过分离对象的构建过程和表示方式,使得构建复杂对象的流程更加清晰和灵活。它非常适合用于那些具有多个可选参数、构建过程复杂或者有不同表示方式的对象。建造者模式的使用可以提高代码的可读性和可维护性,同时减少对象构建中的复杂性。

七、在 Unity 中使用 建造者模式

在 Unity 中,建造者模式可以应用于构建复杂的 3D 场景或对象。假设我们要在场景中建造一个由多个部分组成的复杂建筑(如房子),包括地基、墙壁、屋顶等,这正是使用建造者模式的典型场景。

我们将设计一个简单的场景,展示如何通过建造者模式生成不同配置的房屋。房屋由三部分组成:地基、墙壁和屋顶。通过建造者模式,我们可以灵活地创建不同样式的房屋。

步骤:

  1. 定义一个 House 类,包含地基、墙壁和屋顶。
  2. 定义一个 IHouseBuilder 接口,声明创建房屋各部分的方法。
  3. 创建具体的建造者类,如 WoodenHouseBuilderBrickHouseBuilder,实现不同材料的房屋建造。
  4. 创建一个 Director 类,用于控制房屋的建造流程。
  5. 在 Unity 场景中通过建造者模式生成不同风格的房屋。

1、定义房屋类 House

using UnityEngine;

// 房屋类,包含房屋的各个部分
public class House
{
    public GameObject Foundation { get; set; }
    public GameObject Walls { get; set; }
    public GameObject Roof { get; set; }

    public void ShowHouseInfo()
    {
        Debug.Log("House has been built with Foundation, Walls, and Roof.");
    }
}

2、定义建造者接口 IHouseBuilder

public interface IHouseBuilder
{
    void BuildFoundation();
    void BuildWalls();
    void BuildRoof();
    House GetResult();
}

3、 具体建造者类:木屋建造者 WoodenHouseBuilder 和砖屋建造者 BrickHouseBuilder

3.1 木屋建造者

using UnityEngine;

public class WoodenHouseBuilder : IHouseBuilder
{
    private House house = new House();

    public void BuildFoundation()
    {
        house.Foundation = GameObject.CreatePrimitive(PrimitiveType.Cube);
        house.Foundation.transform.localScale = new Vector3(5, 0.5f, 5);
        house.Foundation.GetComponent<Renderer>().material.color = Color.gray;
    }

    public void BuildWalls()
    {
        house.Walls = GameObject.CreatePrimitive(PrimitiveType.Cube);
        house.Walls.transform.localScale = new Vector3(5, 2.5f, 5);
        house.Walls.transform.position = new Vector3(0, 1.5f, 0);
        house.Walls.GetComponent<Renderer>().material.color = Color.yellow; // 木墙
    }

    public void BuildRoof()
    {
        house.Roof = GameObject.CreatePrimitive(PrimitiveType.Cylinder);
        house.Roof.transform.localScale = new Vector3(5, 1, 5);
        house.Roof.transform.position = new Vector3(0, 3f, 0);
        house.Roof.GetComponent<Renderer>().material.color = Color.red;
    }

    public House GetResult()
    {
        return house;
    }
}

3.2 砖屋建造者

using UnityEngine;

public class BrickHouseBuilder : IHouseBuilder
{
    private House house = new House();

    public void BuildFoundation()
    {
        house.Foundation = GameObject.CreatePrimitive(PrimitiveType.Cube);
        house.Foundation.transform.localScale = new Vector3(5, 0.5f, 5);
        house.Foundation.GetComponent<Renderer>().material.color = Color.gray;
    }

    public void BuildWalls()
    {
        house.Walls = GameObject.CreatePrimitive(PrimitiveType.Cube);
        house.Walls.transform.localScale = new Vector3(5, 2.5f, 5);
        house.Walls.transform.position = new Vector3(0, 1.5f, 0);
        house.Walls.GetComponent<Renderer>().material.color = Color.red; // 砖墙
    }

    public void BuildRoof()
    {
        house.Roof = GameObject.CreatePrimitive(PrimitiveType.Cylinder);
        house.Roof.transform.localScale = new Vector3(5, 1, 5);
        house.Roof.transform.position = new Vector3(0, 3f, 0);
        house.Roof.GetComponent<Renderer>().material.color = Color.green;
    }

    public House GetResult()
    {
        return house;
    }
}

4、定义指挥者 Director

public class Director
{
    private IHouseBuilder houseBuilder;

    public Director(IHouseBuilder builder)
    {
        houseBuilder = builder;
    }

    // 控制房屋的建造流程
    public void ConstructHouse()
    {
        houseBuilder.BuildFoundation();
        houseBuilder.BuildWalls();
        houseBuilder.BuildRoof();
    }

    // 获取构建好的房屋
    public House GetHouse()
    {
        return houseBuilder.GetResult();
    }
}

5、 在 Unity 场景中使用建造者模式

using UnityEngine;

public class HouseBuilderExample : MonoBehaviour
{
    void Start()
    {
        // 创建一个木屋建造者
        IHouseBuilder woodenHouseBuilder = new WoodenHouseBuilder();
        Director director = new Director(woodenHouseBuilder);
        director.ConstructHouse();
        House woodenHouse = director.GetHouse();
        woodenHouse.ShowHouseInfo();

        // 创建一个砖屋建造者
        IHouseBuilder brickHouseBuilder = new BrickHouseBuilder();
        Director brickDirector = new Director(brickHouseBuilder);
        brickDirector.ConstructHouse();
        House brickHouse = brickDirector.GetHouse();
        brickHouse.ShowHouseInfo();
    }
}

6、运行效果

  1. 当运行场景时,系统会根据不同的建造者生成不同的房屋类型(木屋和砖屋),并将房屋的各个部分放置在场景中。
  2. WoodenHouseBuilder 会生成带有灰色地基、黄色墙壁和红色屋顶的木屋。
  3. BrickHouseBuilder 会生成带有灰色地基、红色砖墙和绿色屋顶的砖屋。

通过这两种建造者类的不同实现,我们展示了如何通过建造者模式灵活创建不同类型的复杂对象。

建造者模式允许分步骤创建对象,并且能够复用构建过程生成不同类型的产品。它也提高了代码的可读性和维护性,特别是在对象构建逻辑复杂时。

当对象的构建步骤固定,但最终产品可能有不同的表现方式时,建造者模式尤其适用。在 Unity 中,这种模式可以用于场景中复杂对象(如建筑物、场景)的灵活构建。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2147446.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Element走马灯组件循环播放两个页面是方向不一致

摘要&#xff1a;使用Carousel 走马灯循环播放同一类型的图片、文字等内容&#xff0c;会在循环内容为两组是出现下图 [1]中的现象。本文记录下如何解决 之前项目遇到过一次这个问题&#xff0c;由于indicator-position 指示器不用显示&#xff0c;则判断内容长度为2时&#xf…

恶意Bot流量识别分析实践

1、摘要 随着互联网的发展&#xff0c;自动化工具和脚本&#xff08;Bots&#xff09;的使用越来越普遍。虽然一些善意 Bots 对于网站的正常运行和数据采集至关重要&#xff0c;但恶意 Bots 可能会对网站带来负面影响&#xff0c;如爬取敏感信息、恶意注册、刷流量等。因此&am…

saltstack配置管理

一、saltstack的SSH工作模式 一、salt-ssh介绍 salt-ssh 是 0.17.0 新引入的一个功能&#xff0c;不需要minion对客户端进行管理&#xff0c;也不需要master。salt-ssh 支持salt大部分的功能&#xff1a;如 grains、modules、state 等salt-ssh 没有使用ZeroMQ的通信架构&#…

向日葵好用吗?4款稳定的远程控制软件推荐。

远程控制技术现在已经被应用于很多个领域&#xff0c;像企业办公&#xff0c;远程协助&#xff0c;智能家居&#xff0c;工业控制等等。我们常常会用到的时前两种。而实现远程控制的方式也有多种&#xff0c;但是最方便高效的还是使用第三方软件。我最常使用的是向日葵&#xf…

算法.图论-并查集上

文章目录 1. 并查集介绍2. 并查集的实现2.1 实现逻辑2.2 isSameSet方法2.3 union方法(小挂大优化)2.4 find方法(路径压缩优化) 3. 并查集模板 1. 并查集介绍 定义&#xff1a; 并查集是一种树型的数据结构&#xff0c;用于处理一些不相交集合的合并及查询问题&#xff08;即所…

游戏如何对抗定制挂

近年来&#xff0c;游戏安全对抗强度相比以往更加激烈&#xff0c;具体表现在“定制挂”趋势显著。在近期收集的近万款外挂样本中&#xff0c;定制挂约占比78%&#xff0c;常见的内存修改器、变速器等通用作弊手段占比正在下降。 所谓定制挂&#xff0c;是指针对某款游戏单独开…

SPI接口通信协议浅谈成都自动化开发

沙鸥-成都 1 什么是SPI SPI是串口外设接口的缩写&#xff0c;是一种高速的、全双工、同步的通信协议&#xff0c;是微处理器与外围IC之间常用的一种通讯方式。 SPI是主从式的通信协议&#xff0c;可以一主机一从机通信&#xff0c;也可以一主机多从机通信。 2 SPI的优缺点 SPI接…

【Java版】云HIS系统源码

云HIS系统介绍 云HIS系统是一款满足基层医疗机构各类业务需要的健康云产品。该产品能帮助基层医疗机构完成日常各类业务&#xff0c;提供病患挂号支持、病患问诊、电子病历、开药发药、会员管理、统计查询、医生站和护士站等一系列常规功能&#xff0c;还能与公卫、PACS等各类…

【STM32 HAL库】OLED显示模块

【STM32 HAL库】OLED显示模块 前言理论OLED基本参数OLED基本驱动原理OLED坐标轴 应用CubeMx配置底层函数代码高层封装函数printf显示函数 前言 本文为笔者学习 OLED 的总结&#xff0c;基于keysking的视频内容&#xff0c;如有错误&#xff0c;欢迎指正 理论 OLED基本参数 …

基于224G的超高速以太网端口1.6Tbps 1600G真的来了~

基于224G PAM4 SerDes的1.6T(更激进些的是3.2T&#xff09; 受AI智能算中心的驱动&#xff0c;基于4x112G的400G光模块&#xff0c;和基于8x112G的800G的光模块已经很成熟了&#xff0c;标志就是大家都在降本增效&#xff0c;考虑干掉功耗的DSP&#xff0c;...另外一个标志就是…

关于支持向量机的一份介绍

在这篇文章中&#xff0c;我将介绍与支持向量机有关的东西&#xff0c;我们知道支持向量机主要分两类&#xff0c;就是线性支持向量机和核支持向量机这两种&#xff08;当然还有其他的&#xff0c;如多类支持向量机、 Nu-Support Vector Regression等&#xff09;&#xff0c;因…

AIGC生图基础知识

一、引言 AIGC&#xff0c;即AI-Generated Content&#xff0c;是一种利用大型预训练模型如生成对抗网络&#xff08;GAN&#xff09;、扩散网络&#xff08;Diffusion&#xff09;和语言大模型&#xff08;Transformer&#xff09;等人工智能技术&#xff0c;通过对大量数据进…

Gradio 自定义组件

如何使用 Gradio 自定义组件&#xff0c;Gradio 前端使用 Svelte&#xff0c;后端使用的 Python。如何自定义一个组件呢&#xff1f;Gadio 提供了类似于脚手架的命令&#xff0c;可以生成需要开发组件的前后和后端代码。 创建组件 运行如下命令&#xff0c;gradio 会自动生成…

OBC充电机测试的步骤和规范

一、测试前准备 1. 确认测试环境&#xff1a;确保测试环境的温度、湿度等条件符合设备的工作要求。 2. 检查设备&#xff1a;检查OBC充电机是否完好无损&#xff0c;电源线、插头等是否连接良好&#xff0c;显示屏是否正常显示。 3. 准备工具&#xff1a;准备好电压表、电流…

ubuntu20.04安装cudnn

先登入账号 网址&#xff1a;https://developer.nvidia.com/cudnn 选择ubuntu20.04 x86_64&#xff08;Deb&#xff09; 在下载好文件的文件夹下打开终端 sudo apt-get install zlib1gsudo dpkg -i cudnn-local-repo-${distro}-8.x.x.x_1.0-1_amd64.debsudo cp /var/cudnn-lo…

大数据时代:历史、发展与未来

文章目录 引言1980年&#xff1a;大数据的先声2006年&#xff1a;云计算与大数据的诞生2008年&#xff1a;大数据的科学探索2009年&#xff1a;大数据成为行业热词2011年&#xff1a;大数据的商业价值2013年&#xff1a;世界大数据元年结语 引言 在信息技术飞速发展的今天&…

钢铁焦化水泥超低排的原因

钢铁、焦化和水泥行业实施超低排放的原因&#xff0c;朗观视觉小编建议大家可以从环境保护、产业升级、政策推动以及企业可持续发展等多个方面进行分析。 一、环境保护需求 空气质量改善&#xff1a;钢铁、焦化和水泥行业是传统的高污染行业&#xff0c;其排放的颗粒物、二氧化…

表格HTML

//test.html <!DOCTYPE html> <html lang"en"> <head> <meta charset"UTF-8"> <meta name"viewport" content"widthdevice-width, initial-scale1.0"> <title>表格与CSS分开示例</tit…

【STL】string 基础,应用与操作

string 1.string相关介绍 STL&#xff08;标准模板库&#xff09;中的string容器是C标准库提供的用于处理和操作字符串的类&#xff0c;位于头文件中。std::string提供了比传统的C风格字符串&#xff08;字符数组&#xff09;更方便和安全的功能&#xff0c;具有动态内存管理…

python脚本编译为.so速度对比

有两个好处&#xff1a; 产品代码保护&#xff0c;so文件不可读 计算能力加速&#xff0c;本质上编译过程为python -> c -> so文件&#xff0c;相当于动态语言转换为静态语言&#xff0c;程序执行能力和计算能力有所提升 编译为so文件后比原始python代码执行时间快2ms左…