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

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.mzph.cn/web/53424.shtml

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

相关文章

Android平台Unity3D下如何同时播放多路RTMP|RTSP流?

技术背景 好多开发者&#xff0c;提到希望在Unity的Android头显终端&#xff0c;播放2路以上RTMP或RTSP流&#xff0c;在设备性能一般的情况下&#xff0c;对Unity下的RTMP|RTSP播放器提出了更高的要求。实际上&#xff0c;我们在前几年发布Unity下直播播放模块的时候&#xf…

9.20日学习记录及相关问题解答

部分一 今天看了一本古老的书。学到了一些有关计算机的远古的知识。弥补了一些之前没有意识到的空白点。 原来上个世纪就有AI这个东西了 现阶段的主流模式&#xff0c;在许多年前其实是将来要发展的对象。 B/S指的是客户机/服务器结构模式 C/S是在B/S基础上发展过来的。三层结…

网络安全-LD_PRELOAD,请求劫持

目录 一、环境 二、开始做题 三、总结原理 四、如何防护 一、环境 我们这里用蚁剑自带的靶场第一关来解释 docker制作一下即可 二、开始做题 首先环境内很明显给我们已经写好了webshell 同样我们也可以访问到 我们使用这个蚁剑把这个webshell连上 我们发现命令不能执行&am…

Dockerfile全面指南:从基础到进阶,掌握容器化构建的核心工具

目录 Dockerfile全面指南&#xff1a;从基础到进阶&#xff0c;掌握容器化构建的核心工具 引言 一、什么是 Dockerfile 二、Dockerfile 的基本结构 三、Dockerfile 的常见配置项 1、多阶段构建 (Multi-stage Builds) 2、缓存优化 3、合并 RUN 命令 四、Dockerfile 使用…

从数据仓库到数据中台再到数据飞轮:我了解的数据技术进化史

这里写目录标题 前言数据仓库&#xff1a;数据整合的起点数据中台&#xff1a;数据共享的桥梁数据飞轮&#xff1a;业务与数据的双向驱动结语 前言 在当今这个数据驱动的时代&#xff0c;企业发展离不开对数据的深度挖掘和高效利用。从最初的数据仓库&#xff0c;到后来的数据…

基于SpringBoot+Vue+MySQL的校园一卡通系统

系统展示 用户前台界面 管理员后台界面 系统背景 随着现代社会的快速发展&#xff0c;校园一卡通已成为大学生活中不可或缺的一部分。它不仅承载着校园消费的功能&#xff0c;还集成了学生身份证明、图书馆借阅、门禁系统等多种服务。然而&#xff0c;传统的一卡通管理系统往往…

OpenCL 学习(2)---- OpenCL Platform 和 Device

目录 OpenCL PlatformOpenCL Device参考代码 OpenCL Platform opencl 支持的 Platform 可以使用 clGetPlatformIDs 函数查询&#xff0c;函数原型如下&#xff1a; clGetPlatformIDs(cl_uint /* num_entries */,cl_platform_id * /* platforms */,cl_uint * …

鸿蒙OpenHarmony【轻量系统内核扩展组件(CPU占用率)】子系统开发

基本概念 CPU&#xff08;中央处理器&#xff0c;Central Processing Unit&#xff09;占用率分为系统CPU占用率和任务CPU占用率。 系统CPU占用率&#xff1a;是指周期时间内系统的CPU占用率&#xff0c;用于表示系统一段时间内的闲忙程度&#xff0c;也表示CPU的负载情况。系…

linux中vim编辑器的应用实例

前言 Linux有大量的配置文件&#xff0c;其中编辑一些配置文件&#xff0c;最常用的工具就是 Vim &#xff0c;本文介绍一个实际应用的Vim编辑器开发文档的实例。 Vim是一个类似于Vi的著名的功能强大、高度可定制的文本编辑器&#xff0c;在Vi的基础上改进和增加了很多特性。…

告别枯燥:我开发了一个在电脑桌面上使用弹幕来背单词的软件

前言 在这个快节奏的时代&#xff0c;我们每天都在忙碌中度过&#xff0c;手机虽然方便&#xff0c;但往往难以找到一整块时间来专心背单词。然而&#xff0c;你是否意识到&#xff0c;每天坐在电脑前的时间远比使用手机的时间要长&#xff1f;现在我们来介绍一个新型的学习软…

基于大数据的电子产品需求数据分析系统的设计与实现(Python Vue Flask Mysql)

&#x1f497;博主介绍&#x1f497;&#xff1a;✌在职Java研发工程师、专注于程序设计、源码分享、技术交流、专注于Java技术领域和毕业设计✌ 温馨提示&#xff1a;文末有 CSDN 平台官方提供的老师 Wechat / QQ 名片 :) Java精品实战案例《700套》 2025最新毕业设计选题推荐…

Springboot 文件上传下载相关问题

文章目录 关于Springboot 文件上传下载问题解决方案注意事项文件上传文件下载文件删除文件在线打开在写练习的时候&#xff0c;发现了一些小小的问题&#xff0c;已经在 上述代码中体现。① 代码路径碰到中文的时候&#xff0c;会有乱码&#xff0c;需要转换&#xff08;内容中…

浏览器插件利器--allWebPluginV2.0.0.20-stable版发布

allWebPlugin简介 allWebPlugin中间件是一款为用户提供安全、可靠、便捷的浏览器插件服务的中间件产品&#xff0c;致力于将浏览器插件重新应用到所有浏览器。它将现有ActiveX控件直接嵌入浏览器&#xff0c;实现插件加载、界面显示、接口调用、事件回调等。支持Chrome、Firefo…

我的AI工具箱Tauri版-VideoIntroductionClipCut视频介绍混剪

本教程基于自研的AI工具箱Tauri版进行VideoIntroductionClipCut视频介绍混剪。 本项目为自研的AI工具箱Tauri版中的视频剪辑模块&#xff0c;专注于自动生成视频介绍片段。该模块名为 VideoIntroductionClipCut&#xff0c;用户可以通过该工具快速进行视频的混剪和介绍内容的生…

【网络】高级IO——epoll版本TCP服务器初阶

目录 前言 一&#xff0c;epoll的三个系统调用接口 1.1.epoll_create函数 1.1.1.epoll_create函数干了什么 1.2. epoll_ctl函数 1.2.1.epoll_ctl函数函数干了什么 1.3.epoll_wait函数 1.3.1.epoll_wait到底干了什么 1.4.epoll的工作过程中内核在干什么 二&#xff0c;…

nginx upstream转发连接错误情况研究

本次测试用到3台服务器&#xff1a; 192.168.10.115&#xff1a;转发服务器A 192.168.10.209&#xff1a;upstream下服务器1 192.168.10.210&#xff1a;upstream下服务器2 1台客户端&#xff1a;192.168.10.112 服务器A中nginx主要配置如下&#xff1a; log_format main…

linux下共享内存的3种使用方式

进程是资源封装的单位&#xff0c;内存就是进程所封装的资源的一种。一般情况下&#xff0c;进程间的内存是相互隔离的&#xff0c;也就是说一个进程不能访问另一个进程的内存。如果一个进程想要访问另一个进程的内存&#xff0c;那么必须要进过内核这个桥梁&#xff0c;这就是…

基于SpringBoot+Vue+MySQL的医院信息管理系统

系统展示 用户前台界面 管理员后台界面 系统背景 在当今社会&#xff0c;随着医疗服务需求的不断增长和医疗信息化的快速发展&#xff0c;提升医院管理效率和服务质量成为了医疗行业的核心需求。传统的医院管理模式面临着效率低下、资源分配不均、患者就医体验差等问题。为了应…

Nginx 跨域 + 无法设置 Cookie 解决办法

今天来分享一下关于项目部署上线时怎么解决跨域问题!!! 首先感谢一下大佬的方法,才让这个困扰我很久的问题得以解决!!! 这也是我请教大佬才解决的问题,大佬和我说,这是他耗费两周才解决的问题,我这也是属于前人栽树后人乘凉了,嘿嘿嘿!!! 前端问题 前端没有携带 cookie 导致后端…

uni-app安装插件

1.通过插件市场安装https://ext.dcloud.net.cn 打开HBuilderX编辑器。 点击菜单栏中的“工具”->“插件安装”。 这里会看到已安装插件和安装新插件两个选项卡&#xff0c;点击安装新插件&#xff0c; 能看到一些核心插件&#xff0c;如果所需要的插件在核心插件里面有&…