Unity声音管理系统笔记

需求

  1. 在各个场景可以打开设置面板,设置音乐、UI音效和所有其他音效的音量;

分析

管理器在各个场景都存在,可以:

  1. 不继承MonoBehavior,就不能和AudioSource挂在一个对象,也不能通过拖指定音乐和UI音效AudioSource。如果在场景里建好AudioSource,让它用GameObject.Find()或transform.Find()找AudioSource,那么每个场景都要创建,容易漏掉。所以这条路最好让管理器新建对象,添加AudioSource。
  2. 继承MonoBehavior,可以拖了,每个场景都建一个,同理容易漏建,不是好方案。
  3. 继承MonoBehavior,并且DontDestroyOnLoad。

总之,手动在每个场景创建管理器不是好方案,如果又想拖,那么继承MonoBehavior,DontDestroyOnLoad可能是最好的方案。

可以是管理器有场景里所有AudioSource的引用,并且知道哪个AudioSource是音乐,哪些是音效。初始化和改变音量时把相应的AudioSource的音量写成用户设置的,这里称为"中心改变地方"。

也可以是所有的播放声音函数使用声音管理器封装的函数,里面播放前先把音量设置为用户设置的。这里称为"地方读取中心"。

中心改变地方

管理器怎么分清哪个是音乐,哪些是音效?可以把音乐AudioSource和UI音效AudioSource单独拖。其他通过FindObjectsByType(),再和音乐、UI音效AudioSource对比判断是其他AudioSource。

但是转换场景时要重新获得一遍场景所有的AudioSource。管理器怎么知道什么时候切换场景了?还是需要各场景的管理器调用。

如果还有运行中创建AudioSource,声音管理器又怎么知道?创建AudioSource的脚本通知一下声音管理器?

这样想下去,越来越发现"中心改变地方",设计不可行,或者说越来越变成"地方读取中心"。管理器缺少太多信息,包括AudioSource种类,生命周期。

地方读取中心

需要把项目所有的audioSource.Play()替换成自己封装的方法。先找到一个audioSource.Play(),再对Play()寻找引用,替换audioSource.Play()。

cs 复制代码
public void MyPlaySound(AudioSource audioSource){
        audioSource.volume=DataManager.Instance.settingsData.soundVolume;
        audioSource.Play();
    }

音乐在游戏开始运行时开始播放,需要在初始化、玩家改变音量时设置音量。各AudioSource知道自己是音乐、UI音效还是其他音效,也知道自己什么时候播放,无需关心场景有没有切换。

然后......好像没什么问题了。

总结

总结下来,这种管理系统的设计有点反直觉,看起来是"管理器",其实是各地方组件要求管理器管理自己。

相关推荐
HXR_plume1 小时前
【计算机网络】王道考研笔记整理(1)计算机网络体系结构
网络·笔记·计算机网络
笑鸿的学习笔记2 小时前
qt-C++语法笔记之Stretch与Spacer的关系分析
c++·笔记·qt
巴伦是只猫3 小时前
【机器学习笔记Ⅰ】3 代价函数
人工智能·笔记·机器学习
ZZZS05163 小时前
stack栈练习
c++·笔记·学习·算法·动态规划
AI视觉网奇4 小时前
rag学习笔记
笔记·学习
两水先木示4 小时前
【Unity3D】微信小游戏适配安全区域或胶囊控件(圆圈按钮)水平高度一致方案
unity·微信小游戏·安全区域·ui适配·胶囊控件·safearea
枯萎穿心攻击4 小时前
ECS由浅入深第三节:进阶?System 的行为与复杂交互模式
开发语言·unity·c#·游戏引擎
不绝1915 小时前
怪物机制分析(有限状态机、编辑器可视化、巡逻机制)
网络·游戏·unity·游戏引擎
unicrom_深圳市由你创科技6 小时前
Unity开发如何解决iOS闪退问题
unity·ios·蓝桥杯
teeeeeeemo6 小时前
http和https的区别
开发语言·网络·笔记·网络协议·http·https