Unity 桥接模式(实例详解)

文章目录

      • 示例1:角色与装备系统
      • 示例2:UI控件库
      • 示例3:渲染引擎模块
      • 示例4:AI决策树算法
      • 示例5:物理模拟引擎

在Unity游戏开发中,桥接模式(Bridge Pattern)是一种设计模式,它用于将抽象部分与其实现部分分离,从而允许它们独立变化。这种模式适用于以下场景:

  1. 当一个类有多个维度的变化,并且希望这些变化可以独立扩展而不影响彼此时。
  2. 如果直接使用继承来处理多维度变化会导致类爆炸问题,即需要创建大量子类。

桥接模式的组成部分:

  • 抽象(Abstraction):定义了抽象接口,并包含对实现部分(Implementor)的操作。在Unity中,这可能是某个基础组件或接口,它提供了通用的功能描述,但不关心具体的实现细节。

例如:

public interface ICharacterControl
{void Move();void Attack();void SetWeapon(IWeapon weapon);
}public interface IWeapon
{void Fire();void Reload();
}
  • 提炼的抽象(Refined Abstraction):继承自抽象类的具体角色,它可能会添加更多特定的行为,同时调用实现部分的方法。
public class Soldier : MonoBehaviour, ICharacterControl
{private IWeapon _currentWeapon;public void Move(){// 实现移动逻辑}public void Attack(){if (_currentWeapon != null)_currentWeapon.Fire();}public void SetWeapon(IWeapon weapon){_currentWeapon = weapon;}
}
  • 实现(Implementor):定义了实现接口,这个接口通常代表了可变的部分,如不同的武器类型。
public abstract class WeaponBase : IWeapon
{public abstract void Fire();public abstract void Reload();
}public class Pistol : WeaponBase
{public override void Fire(){Debug.Log("Pistol fires!");}public override void Reload(){Debug.Log("Pistol is reloading...");}
}public class Shotgun : WeaponBase
{public override void Fire(){Debug.Log("Shotgun fires!");}public override void Reload(){Debug.Log("Shotgun is reloading...");}
}

五个实例说明:

  1. 角色和装备系统:游戏角色可以有不同的移动方式(跑、走、跳等)以及不同的武器(枪、剑、魔法等),通过桥接模式可以使角色类型和武器类型相互独立地扩展。

  2. 渲染引擎模块:抽象层为渲染器接口,具体实现包括不同的渲染技术(如Direct3D、OpenGL、Vulkan等)。不论游戏采用哪种渲染技术,上层的游戏对象渲染逻辑保持不变。

  3. UI控件库:抽象层定义了一系列UI控件(按钮、文本框、滑块等)的公共行为,而具体实现可能基于不同的图形API或框架(Unity UI、NGUI、TextMeshPro等)。

  4. AI决策树算法:抽象层是决策树接口,不同类型的AI实体可以根据需求选择不同的决策树实现(简单状态机、有限状态机、蒙特卡洛搜索树等)。

  5. 物理模拟引擎:抽象层提供物理模拟功能,如碰撞检测、刚体运动等,而具体实现可以切换为不同的物理引擎(Unity内置物理引擎、PhysX、Box2D等)。

以下是在Unity中应用桥接模式的五个不同场景的简化代码示例:

示例1:角色与装备系统

// 抽象部分 - 角色接口
public interface ICharacter
{void Move();void ChangeWeapon(IWeapon weapon);
}// 实现部分 - 武器接口
public interface IWeapon
{void Use();void Reload();
}// 具体抽象 - 战士类,继承自ICharacter
public class Warrior : MonoBehaviour, ICharacter
{private IWeapon currentWeapon;public void Move(){// 移动逻辑}public void ChangeWeapon(IWeapon weapon){currentWeapon = weapon;}public void Attack(){if (currentWeapon != null)currentWeapon.Use();}
}// 实现部分的具体类 - 短剑和长弓
public class ShortSword : IWeapon
{public void Use(){Debug.Log("Short sword is used for attack!");}public void Reload(){// 无需重新加载}
}public class LongBow : IWeapon
{public void Use(){Debug.Log("Long bow fires an arrow!");}public void Reload(){Debug.Log("Reloading the long bow...");}
}// 在游戏运行时切换武器
var warrior = GetComponent<Warrior>();
warrior.ChangeWeapon(new ShortSword());
warrior.Attack(); // 输出:"Short sword is used for attack!"
warrior.ChangeWeapon(new LongBow());
warrior.Attack(); // 输出:"Long bow fires an arrow!"

示例2:UI控件库

// 抽象部分 - UI控件接口
public interface IUIControl
{void Render();void SetText(string text);
}// 实现部分 - 不同UI框架下的文本框实现
public abstract class TextControlBase : IUIControl
{public abstract void Render();public abstract void SetText(string text);
}public class UnityUITextControl : TextControlBase
{public UnityEngine.UI.Text unityText; // Unity UI组件public override void Render(){// 使用Unity UI渲染文本}public override void SetText(string text){unityText.text = text;}
}public class NGUITextControl : TextControlBase
{// 假设NGUI有对应的文本组件引用public override void Render(){// 使用NGUI渲染文本}public override void SetText(string text){// 设置NGUI文本内容}
}// 在游戏运行时创建不同的UI文本框实例
var uiText = new UnityUITextControl();
uiText.SetText("Hello from Unity UI");
uiText.Render();var nguiText = new NGUITextControl();
nguiText.SetText("Hello from NGUI");
nguiText.Render();

示例3:渲染引擎模块

// 抽象部分 - 渲染器接口
public interface IRenderer
{void RenderScene(GameObject scene);
}// 实现部分 - 不同的渲染引擎实现
public class DirectXRenderer : IRenderer
{public void RenderScene(GameObject scene){// 使用DirectX渲染场景Debug.Log("Rendering scene with DirectX...");}
}public class OpenGLRenderer : IRenderer
{public void RenderScene(GameObject scene){// 使用OpenGL渲染场景Debug.Log("Rendering scene with OpenGL...");}
}// 游戏中的场景管理器调用渲染逻辑
public class SceneManager
{private IRenderer _renderer;public SceneManager(IRenderer renderer){_renderer = renderer;}public void RenderCurrentScene(){_renderer.RenderScene(currentScene);}
}// 在游戏启动时根据需要选择渲染引擎
var directXRenderer = new DirectXRenderer();
var sceneManager = new SceneManager(directXRenderer);
sceneManager.RenderCurrentScene(); // 输出:"Rendering scene with DirectX..."// 如果需要切换到OpenGL,只需更改渲染器实例
var openGLRenderer = new OpenGLRenderer();
sceneManager._renderer = openGLRenderer;
sceneManager.RenderCurrentScene(); // 输出:"Rendering scene with OpenGL..."

示例4:AI决策树算法

// 抽象部分 - 决策树接口
public interface IDecisionTree
{Action Decide(AIState state);
}// 实现部分 - 简单状态机和有限状态机
public class SimpleStateMachine : IDecisionTree
{public Action Decide(AIState state){// 根据简单状态机决定动作Debug.Log($"Decided action based on Simple State Machine: {state}");return () => { /* 执行具体动作 */ };}
}public class FiniteStateMachine : IDecisionTree
{public Action Decide(AIState state){// 根据有限状态机决定动作Debug.Log($"Decided action based on Finite State Machine: {state}");return () => { /* 执行具体动作 */ };}
}// AI角色类使用决策树接口
public class AIBot
{private IDecisionTree _decisionTree;public AIBot(IDecisionTree decisionTree){_decisionTree = decisionTree;}public void TakeAction(AIState currentState){var action = _decisionTree.Decide(currentState);action?.Invoke();}
}// 在游戏运行时选择不同的决策树
var simpleBot = new AIBot(new SimpleStateMachine());
simpleBot.TakeAction(AIState.Idle);var fsmBot = new AIBot(new FiniteStateMachine());
fsmBot.TakeAction(AIState.Patrolling);

示例5:物理模拟引擎

// 抽象部分 - 物理引擎接口
public interface IPhysicsEngine
{void Simulate();Rigidbody CreateRigidbody(GameObject obj);
}// 实现部分 - Unity内置物理引擎和Box2D
public class UnityPhysicsEngine : IPhysicsEngine
{public void Simulate(){// 调用Unity内置物理引擎进行模拟Physics.Simulate(Time.fixedDeltaTime);}public Rigidbody CreateRigidbody(GameObject obj){return obj.AddComponent<Rigidbody>();}
}public class Box2DPhysicsEngine : IPhysicsEngine
{// 假设有一个用于与Box2D交互的封装类Box2DBodypublic void Simulate(){// 使用Box2D进行物理模拟// ...}public Rigidbody CreateRigidbody(GameObject obj){// 创建一个Box2D对应的刚体组件并附加到对象上// 返回Box2DBody的引用或包装对象return null; // 这里简化处理,实际需要实现创建Box2D刚体}
}// 游戏中的物理世界管理器使用物理引擎接口
public class PhysicsWorldManager
{private IPhysicsEngine _physicsEngine;public PhysicsWorldManager(IPhysicsEngine physicsEngine){_physicsEngine = physicsEngine;}public void UpdatePhysics(){_physicsEngine.Simulate();}public Rigidbody AddRigidbody(GameObject obj){return _physicsEngine.CreateRigidbody(obj);}
}// 在游戏启动时选择物理引擎
var unityPhysics = new UnityPhysicsEngine();
var physicsManager = new PhysicsWorldManager(unityPhysics);
// ...// 如果要切换到Box2D
var box2DPhysics = new Box2DPhysicsEngine();
physicsManager._physicsEngine = box2DPhysics;

python推荐学习汇总连接:
50个开发必备的Python经典脚本(1-10)

50个开发必备的Python经典脚本(11-20)

50个开发必备的Python经典脚本(21-30)

50个开发必备的Python经典脚本(31-40)

50个开发必备的Python经典脚本(41-50)
————————————————

​最后我们放松一下眼睛
在这里插入图片描述

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

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

相关文章

PN532测试工具

PN532测试工具&#xff0c;可以读写卡&#xff0c;修改数据&#xff0c;格式化清卡 读写UID卡&#xff0c;CUID卡&#xff0c;锁UFUID卡 如下图&#xff0c;软件简单易用 可以对UID卡、CUID卡&#xff0c;FUID卡、UFUID卡读卡号&#xff0c;修改卡号 操作简单易用 软件下载地址…

thinkphp5实战之phpstudy v8环境搭建,解决Not Found找不到路径问题

引言 thinkphp以快速、简约的大道至简的思想广受欢迎&#xff0c;适合开发小型项目。本地环境下&#xff0c;phpstudy v8是一款比较优秀的集成环境软件。部署完项目后&#xff0c;访问的时候傻眼&#xff0c;报错。 解决方案 不要慌&#xff0c;这个是伪静态的原因。选择apach…

图卷积网络(GCN)

本文主要分为两部分&#xff0c;第一部分介绍什么是GCN&#xff0c;第二部分将进行详细的数学推导。 一、什么是GCN 1、GCN 概述 本文讲的GCN 来源于论文&#xff1a;SEMI-SUPERVISED CLASSIFICATION WITH GRAPH CONVOLUTIONAL NETWORKS&#xff0c;这是在GCN领域最经典的论文…

两个指针之间的运算

1 、两个指针之间可以做减法运算&#xff0c;但不能做加法运算。 2 、两指针变量相减所得之差是两个指针所指数组元素之间相差的元素个数。

【JS逆向学习】某壁纸下载(ast混淆)

逆向目标 目标网址&#xff1a;https://bz.zzzmh.cn/index逆向接口一&#xff1a;https://api.zzzmh.cn/bz/v3/getData逆向接口二&#xff1a;https://cdn2.zzzmh.cn/wallpaper/origin/0d7d8d691e644989b72ddda5f695aca2.jpg?response-content-dispositionattachment&aut…

大模型笔记【3】 gem5 运行模型框架LLama

一 LLama.cpp LLama.cpp 支持x86&#xff0c;arm&#xff0c;gpu的编译。 1. github 下载llama.cpp https://github.com/ggerganov/llama.cpp.git 2. gem5支持arm架构比较好&#xff0c;所以我们使用编译LLama.cpp。 以下是我对Makefile的修改 开始编译&#xff1a; make UNAME…

AI搜索引擎Perplexity来了,谷歌等老牌搜索引擎或许会有新的威胁?

Perplexity AI 是一家 AI 搜索初创公司&#xff0c;它通过结合内容索引技术和大型语言模型的推理能力&#xff0c;提供更便捷和高效的搜索体验。另外&#xff0c;最近很火的小兔子Rabbit R1硬件AI设备中的搜索功能正是这家公司的杰作。在短短一年半的时间里&#xff0c;一个企业…

【第十五课】数据结构:堆 (“堆”的介绍+主要操作 / acwing-838堆排序 / c++代码 )

目录 关于堆的一些知识的回顾 数据结构&#xff1a;堆的特点 "down" 和 "up"&#xff1a;维护堆的性质 down up 数据结构&#xff1a;堆的主要操作 acwing-838堆排序 代码如下 时间复杂度分析 确实是在写的过程中频繁回顾了很多关于树的知识&…

性能优化-OpenCL 介绍

「发表于知乎专栏《移动端算法优化》」 本文首先对 GPU 进行了概述&#xff0c;然后着重地对移动端的 GPU 进行了分析&#xff0c;随后我们又详细地介绍了 OpenCL 的背景知识和 OpenCL 的四大编程模型。希望能帮助大家更好地进行移动端高性能代码的开发。 &#x1f3ac;个人简介…

【Android】细数Linux和Android系统中的伪文件系统

文章目录 前言Linux伪文件系统cgroupfsLinux的cgroupsAndroid的cgroups debugfsfunctionfs(/dev/usb-ffs/adb)functionfs 的引入sysfs是什么 procfs(/proc)pstore(/sys/fs/pstore)selinuxfs(/sys/fs/selinux)sysfs(/sys)参考 前言 做了好些年Android开发&#xff0c;你了解过L…

RFID标签是什么?该技术有哪些应用领域?

射频识别&#xff08;RFID&#xff09;技术利用电磁场&#xff0c;自动识别和跟踪附在物体上的标签&#xff0c;其中&#xff0c;近场通信&#xff08;NFC&#xff09;是一种基于短距离RFID高频技术的标准&#xff0c;支持13.56 MHz的频率。 NFC技术在现今的产品中应用广泛&am…

【心得】java从CC1链入门CC链个人笔记

来劲了&#xff0c;感觉离真正的CTF又近了一步。 本文仅从一个萌新的角度去谈&#xff0c;如有纰漏&#xff0c;纯属蒟蒻。 目录 CC链概念 CC链学习前置知识 CC1链 Version1 Version2 Version3 CC链概念 CC链 Commons Collections apache组织发布的开源库 里面主要对…