- •Предисловие
- •Введение
- •Благодарности
- •О книге
- •Перспективы
- •Условные обозначения, требования и доступные для скачивания данные
- •Автор в Интернете
- •Об авторе
- •Глава 1. Знакомство с Unity
- •1.1. Достоинства Unity
- •1.1.1. Сильные стороны и преимущества Unity
- •1.1.2. Недостатки, о которых нужно знать
- •1.1.3. Примеры игр на основе Unity
- •1.2. Как работать с Unity
- •1.2.1. Вкладка Scene, вкладка Game и панель инструментов
- •1.2.2. Работа с мышью и клавиатурой
- •1.2.3. Вкладка Hierarchy и панель Inspector
- •1.2.4. Вкладки Project и Console
- •1.3. Готовимся программировать в Unity
- •1.3.1. Запуск кода в Unity: компоненты сценария
- •1.3.2. Программа MonoDevelop — межплатформенная среда разработки
- •1.4. Заключение
- •Глава 2. Создание 3D-ролика
- •2.1. Подготовка…
- •2.1.1. Планирование проекта
- •2.1.2. Трехмерное координатное пространство
- •2.2. Начало проекта: размещение объектов
- •2.2.1. Декорации: пол, внешние и внутренние стены
- •2.2.2. Источники света и камеры
- •2.2.3. Коллайдер и точка наблюдения игрока
- •2.3. Двигаем объекты: сценарий, активирующий преобразования
- •2.3.1. Схема программирования движения
- •2.3.2. Написание кода
- •2.3.3. Локальные и глобальные координаты
- •2.4. Компонент сценария для осмотра сцены: MouseLook
- •2.4.1. Горизонтальное вращение, следящее за указателем мыши
- •2.4.2. Поворот по вертикали с ограничениями
- •2.4.3. Одновременные горизонтальное и вертикальное вращения
- •2.5. Компонент для клавиатурного ввода
- •2.5.1. Реакция на нажатие клавиш
- •2.5.2. Независимая от скорости работы компьютера скорость перемещений
- •2.5.4. Ходить, а не летать
- •2.6. Заключение
- •3.1. Стрельба путем бросания лучей
- •3.1.1. Что такое бросание лучей?
- •3.1.2. Имитация стрельбы командой ScreenPointToRay
- •3.1.3. Добавление визуальных индикаторов для прицеливания и попаданий
- •3.2. Создаем активные цели
- •3.2.1. Определяем точку попадания
- •3.2.2. Уведомляем цель о попадании
- •3.3. Базовый искусственный интеллект для перемещения по сцене
- •3.3.1. Диаграмма работы базового искусственного интеллекта
- •3.3.2. «Поиск» препятствий методом бросания лучей
- •3.3.3. Слежение за состоянием персонажа
- •3.4.1. Что такое шаблон экземпляров?
- •3.4.2. Создание шаблона врага
- •3.4.3. Экземпляры невидимого компонента SceneController
- •3.5. Стрельба путем создания экземпляров
- •3.5.1. Шаблон снаряда
- •3.5.2. Стрельба и столкновение с целью
- •3.5.3. Повреждение игрока
- •3.6. Заключение
- •Глава 4. Работа с графикой
- •4.1. Основные сведения о графических ресурсах
- •4.2. Создание геометрической модели сцены
- •4.2.1. Назначение геометрической модели
- •4.2.2. Рисуем план уровня
- •4.2.3. Расставляем примитивы в соответствии с планом
- •4.3. Наложение текстур
- •4.3.1. Выбор формата файла
- •4.3.2. Импорт файла изображения
- •4.3.3. Назначение текстуры
- •4.4. Создание неба с помощью текстур
- •4.4.1. Что такое скайбокс?
- •4.4.2. Создание нового материала для скайбокса
- •4.5. Собственные трехмерные модели
- •4.5.1. Выбор формата файла
- •4.5.2. Экспорт и импорт модели
- •4.6. Системы частиц
- •4.6.1. Редактирование параметров эффекта
- •4.6.2. Новая текстура для пламени
- •4.6.3. Присоединение эффектов частиц к трехмерным объектам
- •4.7. Заключение
- •5.1. Подготовка к работе с двухмерной графикой
- •5.1.1. Подготовка проекта
- •5.1.2. Отображение двухмерных изображений (спрайтов)
- •5.1.3. Переключение камеры в режим 2D
- •5.2. Создание карт и превращение их в интерактивные объекты
- •5.2.1. Создание объекта из спрайтов
- •5.2.2. Код ввода с помощью мыши
- •5.2.3. Открытие карты по щелчку
- •5.3. Отображение различных карт
- •5.3.1. Программная загрузка изображений
- •5.3.3. Создание экземпляров карт
- •5.3.4. Тасуем карты
- •5.4. Совпадения и подсчет очков
- •5.4.1. Сохранение и сравнение открытых карт
- •5.4.2. Скрытие несовпадающих карт
- •5.4.3. Текстовое отображение счета
- •5.5. Кнопка Restart
- •5.5.1. Добавление к компоненту UIButton метода SendMessage
- •5.5.2. Вызов метода LoadLevel в сценарии SceneController
- •5.6. Заключение
- •Глава 6. Двухмерный GUI для трехмерной игры
- •6.1. Перед тем как писать код…
- •6.1.1. IMGUI или усовершенствованный 2D-интерфейс?
- •6.1.2. Выбор компоновки
- •6.1.3. Импорт изображений UI
- •6.2. Настройка GUI
- •6.2.1. Холст для интерфейса
- •6.2.2. Кнопки, изображения и текстовые подписи
- •6.2.3. Управление положением элементов UI
- •6.3. Программирование интерактивного UI
- •6.3.1. Программирование невидимого объекта UIController
- •6.3.2. Создание всплывающего окна
- •6.3.3. Задание значений с помощью ползунка и поля ввода
- •6.4. Обновление игры в ответ на события
- •6.4.1. Интегрирование системы сообщений
- •6.4.2. Рассылка и слушание сообщений сцены
- •6.4.3. Рассылка и слушание сообщений проекционного дисплея
- •6.5. Заключение
- •7.1. Корректировка положения камеры
- •7.1.1. Импорт персонажа
- •7.1.2. Добавление в сцену теней
- •7.1.3. Облет камеры вокруг персонажа
- •7.2. Элементы управления движением, связанные с камерой
- •7.2.1. Поворот персонажа лицом в направлении движения
- •7.2.2. Движение вперед в выбранном направлении
- •7.3. Выполнение прыжков
- •7.3.1. Добавление вертикальной скорости и ускорения
- •7.3.2. Распознавание поверхности с учетом краев и склонов
- •7.4. Анимация персонажа
- •7.4.1. Создание анимационных клипов для импортированной модели
- •7.4.2. Создание контроллера для анимационных клипов
- •7.4.3. Код, управляющий контроллером-аниматором
- •7.5. Заключение
- •8.1. Создание дверей и других устройств
- •8.1.1. Открывание и закрывание дверей
- •8.1.2. Проверка расстояния и направления перед открытием двери
- •8.1.3. Управление меняющим цвет монитором
- •8.2. Взаимодействие с объектами путем столкновений
- •8.2.1. Столкновение с препятствиями, обладающими физическими свойствами
- •8.2.2. Управление дверью с помощью триггера
- •8.2.3. Сбор разбросанных по игровому уровню элементов
- •8.3. Управление инвентаризационными данными и состоянием игры
- •8.3.1. Настраиваем диспетчеры игрока и инвентаря
- •8.3.2. Программирование диспетчеров
- •8.3.3. Сохранение инвентаря в виде коллекции: списки и словари
- •8.4. Интерфейс для использования и подготовки элементов
- •8.4.1. Отображение элементов инвентаря в UI
- •8.4.2. Подготовка ключа для открытия двери
- •8.4.3. Восстановление здоровья персонажа
- •8.5. Заключение
- •9.1. Создание натурной сцены
- •9.1.1. Генерирование неба с помощью скайбокса
- •9.1.2. Настройка управляемой кодом атмосферы
- •9.2. Скачивание сводки погоды из Интернета
- •9.2.1. Запрос веб-данных через сопрограмму
- •9.2.2. Парсинг текста в формате XML
- •9.2.3. Парсинг текста в формате JSON
- •9.2.4. Изменение вида сцены на базе данных о погоде
- •9.3. Добавление рекламного щита
- •9.3.1. Загрузка изображений из Интернета
- •9.3.2. Вывод изображения на щите
- •9.3.3. Кэширование скачанного изображения
- •9.4. Отправка данных на веб-сервер
- •9.4.1. Слежение за погодой: отправка запросов POST
- •9.4.2. Серверный код в PHP-сценарии
- •9.5. Заключение
- •Глава 10. Звуковые эффекты и музыка
- •10.1. Импорт звуковых эффектов
- •10.1.1. Поддерживаемые форматы файлов
- •10.1.2. Импорт аудиофайлов
- •10.2. Воспроизведение звуковых эффектов
- •10.2.1. Система воспроизведения: клипы, источник, подписчик
- •10.2.2. Присваивание зацикленного звука
- •10.2.3. Активация звуковых эффектов из кода
- •10.3. Интерфейс управления звуком
- •10.3.1. Настройка центрального диспетчера управления звуком
- •10.3.2. UI для управления громкостью
- •10.3.3. Воспроизведение звуков UI
- •10.4. Фоновая музыка
- •10.4.1. Воспроизведение музыкальных циклов
- •10.4.2. Отдельная регулировка громкости
- •10.4.3. Переход между песнями
- •10.5. Заключение
- •Глава 11. Объединение фрагментов в готовую игру
- •11.1. Построение ролевого боевика изменением назначения проектов
- •11.1.1. Сборка ресурсов и кода из разных проектов
- •11.1.2. Элементы наведения и щелчка
- •11.1.3. Замена старого GUI новым
- •11.2. Разработка общей игровой структуры
- •11.2.1. Управление ходом миссии и набором уровней
- •11.2.2. Завершение уровня
- •11.2.3. Проигрыш уровня
- •11.3. Обработка хода игры
- •11.3.1. Сохранение и загрузка достижений игрока
- •11.3.2. Победа в игре при прохождении всех уровней
- •11.4. Заключение
- •Глава 12. Развертывание игр на устройствах игроков
- •12.1. Создание приложений для настольных компьютеров: Windows, Mac и Linux
- •12.1.1. Построение приложения
- •12.1.2. Настройки проигрывателя: имя и значок приложения
- •12.1.3. Компиляция в зависимости от платформы
- •12.2. Создание игр для Интернета
- •12.2.1. Проигрыватель Unity и HTML5/WebGL
- •12.2.2. Создание файла Unity и тестовой веб-страницы
- •12.2.3. Обмен данными с JavaScript в браузере
- •12.3. Сборки для мобильных устройств: iOS и Android
- •12.3.1. Настройка инструментов сборки
- •12.3.2. Сжатие текстур
- •12.3.3. Разработка подключаемых модулей
- •12.4. Заключение
- •Приложение А. Перемещение по сцене и клавиатурные комбинации
- •А.1. Навигация с помощью мыши
- •А.2. Распространенные клавиатурные комбинации
- •Б.1. Инструменты программирования
- •Б.1.1. Visual Studio
- •Б.1.2. Xcode
- •Б.1.3. Android SDK
- •Б.1.4. SVN, Git или Mercurial
- •Б.2. Приложения для работы с трехмерной графикой
- •Б.2.1. Maya
- •Б.2.3. Blender
- •Б.3. Редакторы двухмерной графики
- •Б.3.1. Photoshop
- •Б.3.2. GIMP
- •Б.3.3. TexturePacker
- •Б.4. Звуковое программное обеспечение
- •Б.4.1. Pro Tools
- •Б.4.2. Audacity
- •Приложение В. Моделирование скамейки в программе Blender
- •В.1. Создание сеточной геометрии
- •В.2. Назначение материала
8.3. Управление инвентаризационными данными и состоянием игры 199
ВНИМАНИЕ Метод Destroy() должен вызываться для параметра this.gameObject, а не this! Не путайте эти вещи; ключевое слово this ссылается только на компонент сценария, в то время как выражение this.gameObject ссылается на объект, к которому присоединен сценарий.
Добавленная в код переменная должна появиться на панели Inspector. Введите в соответствующее поле имя, чтобы идентифицировать элемент; я выбрал для первого элемента имя energy. Затем создайте несколько копий элемента и поменяйте их имена; я использовал имена ore, health и key (точность написания имен крайне важна, так как впоследствии они появятся в коде). Заодно назначьте каждому элементу собственный материал, чтобы они имели разные цвета. Я выбрал голубой цвет для элемента energy, темно-серый — для элемента ore, розовый — для элемента health и желтый — для элемента key.
СОВЕТ Вместо имен, как в нашем случае, в более сложных играх у элементов зачастую имеются идентификаторы, используемые для поиска дальнейшей информации. Например, элементу может быть назначен идентификатор 301, который совпадает с определенным отображаемым именем, изображением, описанием и т. п.
Теперь превратите все элементы в шаблоны экземпляров, чтобы упростить их добавление в игровой уровень. В главе 3 вы узнали, что эта операция осуществляется перетаскиванием объектов со вкладки Hierarchy на вкладку Project.
ПРИМЕЧАНИЕ После этой операции имя объекта на панели Hierarchy выделяется синим цветом — это признак объектов, которые являются экземплярами данного шаблона экземпляров. Щелкните правой кнопкой мыши на имени такого объекта и выберите команду Select Prefab для выделения шаблона, экземпляром которого является объект.
Перетаскивая шаблоны, расположите элементы на открытых участках уровня; для тестирования создавайте по несколько экземпляров одного элемента. Запустите игру и «соберите» элементы. Все выглядит здорово, но пока ничего не происходит. Давайте начнем отслеживать собранные элементы; для этого нам нужно создать структуру кода, отвечающую за управление инвентарем.
8.3. Управление инвентаризационными данными и состоянием игры
Теперь, когда мы запрограммировали процедуру сбора элементов, требуется фоновый диспетчер данных (напоминающий веб-форму с ячейками) для игрового инвентаря. Код, который мы напишем, напоминает лежащую в основе многих веб-приложений MVC-архитектуру. Преимуществом такой архитектуры является разделение отображаемых на экране объектов и хранения данных, что упрощает эксперименты и итеративную разработку. Даже в случае сложноорганизованных данных и/или отображения объектов, редактирование части приложения не влияет на остальные его фрагменты.
В играх используются различные варианты таких структур. В каждой игре свои требования к управлению данными, поэтому правила вида «в основе любой игры должен
200 Глава 8. Добавление в игру интерактивных устройств и элементов
лежать вот такой паттерн проектирования» не имеют смысла. Разговор о таких вещах на слишком раннем этапе вообще приведет к снижению продуктивности, так как люди начнут подгонять свое будущее творение под вид рекомендованного паттерна.
В то же время, например, в ролевой игре требования к управлению данными высоки, поэтому, скорее всего, реализовать MVC-архитектуру целесообразно. А в головоломках подобных требований нет, и построение сложной несвязной структуры диспетчеров данных — напрасная трата времени и сил. Состояние такой игры без проблем отслеживается связанными со сценой объектами-контроллерами (более того, вы делали это в предыдущих главах).
Но сейчас перед нами стоит задача по управлению инвентарем игрока. Давайте запрограммируем соответствующую структуру кода.
8.3.1. Настраиваем диспетчеры игрока и инвентаря
Нам нужно разбить управление данными на отдельные хорошо продуманные модули, у каждого из которых своя зона ответственности. Для управления состоянием игрока (например, его здоровьем) мы создадим модуль PlayerManager, а для управления инвентарем — модуль InventoryManager. Диспетчеры данных будут вести себя как модели в MVC-архитектуре; контроллером в большинстве сцен послужит невидимый объект (в данном случае он не требовался, но вспомните объект SceneController из предыдущих глав), остальная же часть сцены выступит аналогом представления.
Создадим мы и расположенный на более высоком уровне «диспетчер диспетчеров», предназначенный для слежения за отдельными модулями. Кроме хранения списка диспетчеров, этот высокоуровневый диспетчер будет контролировать их жизненный цикл, в частности, занимаясь их инициализацией в начальный момент. Доступ всех прочих игровых сценариев к этим централизованным модулям должен осуществляться через главный диспетчер. В частности, для связи с нужным модулем остальной код может использовать ряд статических свойств в главном диспетчере.
ДОСТУП К ЦЕНТРАЛИЗОВАННЫМ МОДУЛЯМ СОВМЕСТНОГО ИСПОЛЬЗОВАНИЯ
За прошедшие годы для решения проблемы соединения частей программы с централизованными модулями общего доступа было создано множество паттернов проектирования. Кпримеру, паттерн одиночки (singleton) упоминался еще в первой книге «Банды четырех».
Однако этот паттерн разонравился многим разработчикам ПО, предпочитающим пользоваться такими альтернативами, как локатор служб (service locator) и инъекция зависимостей (dependency injection). Мой код представляет собой компромисс между простотой статических переменных и гибкостью локатора служб.
Представленный вашему вниманию проект объединяет простой в применении код с возможностью менять различные модули. Например, запрашивая InventoryManager с использованием паттерна одиночки, мы всегда будем ссылаться на один и тот же класс, плотно связывая код с этим классом; в то же время, запрос инвентаря через локатор служб позволяет вернуть как InventoryManager, так и DifferentInventoryManager. Иногда нам требуется возможность переключаться между слегка различающимися версиями одного модуля (например, при развертывании игры на разных платформах).
Чтобы главный диспетчер ссылался на другие модули одним и тем же способом, все эти модули должны унаследовать свойства от общей основы. Мы обеспечим
8.3. Управление инвентаризационными данными и состоянием игры 201
соблюдение этого условия с помощью интерфейса; многие языки программирования (в том числе C#) позволяют задавать своего рода сценарий, которому должны следовать остальные классы. Как PlayerManager, так и InventoryManager будут реализовывать общий интерфейс (он будет называться IGameManager), и основной объект Managers сможет воспринимать их как тип IGameManager. Иллюстрация этой схемы показана на рис. 8.5.
|
|
|
PlayerManager |
|
|
Д |
|
|
|
|
|
|
|
|
|
|
|
• static Player |
|
|
IGameManager |
|
|
|
|||||
• static Inventory |
|
|
|
|
InventoryManager |
|
|
|
|
||
• List<IGameManager> |
|
|
|
||
|
|
||||
|
|
|
|
|
IGameManager |
|
|
|
|
|
|
|
Рис. 8.5. Схема связи модулей |
|
|||
|
|
Кстати, хотя описанная мной архитектура состоит из невидимых модулей, существующих в фоновом режиме, для запуска этого кода Unity все равно требуются сценарии, связанные с объектами сцены. И как в случае с привязанными к сцене контроллерами в предыдущих проектах, мы создадим пустой объект GameObject, который будет связывать эти данные с диспетчерами.
8.3.2. Программирование диспетчеров
Итак, теперь, когда я объяснил все концепции, которыми мы будем пользоваться, пришло время написать код. Первым делом создайте новый сценарий IGameManager и скопируйте в него код следующего листинга.
Листинг 8.9. Базовый интерфейс, который будут реализовывать диспетчеры данных
public interface IGameManager {
ManagerStatus status {get;} ¬ Это перечисление, которое нам нужно определить.
void Startup();
}
Этот файл практически не содержит кода. Обратите внимание, что там отсутствует даже наследование от класса MonoBehaviour; сам по себе интерфейс ничего не делает и существует только затем, чтобы задавать структуру для других классов. Этот интерфейс объявляет одно свойство (переменную, у которой есть функция чтения) и один метод; как свойство, так и метод должны присутствовать у реализующего данный интерфейс класса. Свойство status сообщает остальной части кода, завершил ли модуль инициализацию. Метод Startup() предназначен для обработки процесса инициализации диспетчера, поэтому именно в нем выполняются все связанные с этим процессом задания, а функция задает состояние диспетчера.
202 Глава 8. Добавление в игру интерактивных устройств и элементов
Обратите внимание, что свойство относится к типу ManagerStatus; это пока еще не написанное нами перечисление. Создайте сценарий ManagerStatus.cs и скопируйте в него код из следующего листинга.
Листинг 8.10. ManagerStatus: возможные состояния для состояния IGameManager
public enum ManagerStatus { Shutdown,
Initializing, Started
}
Это еще один файл с практически отсутствующим кодом. На этот раз мы перечисляем возможные состояния диспетчеров, принудительно устанавливая для свойства status одно из указанных значений.
Теперь, когда у нас есть интерфейс IGameManager, мы можем реализовывать его в других сценариях. Листинги 8.11 и 8.12 содержат код сценариев PlayerManager
и InventoryManager.
Листинг 8.11. InventoryManager
using UnityEngine;
using System.Collections;
using System.Collections.Generic; ¬ Импорт новых структур данных (используемых в листинге 8.14).
public class InventoryManager : MonoBehaviour, IGameManager { public ManagerStatus status {get; private set;} ¬
public void Startup() { Сюда идут все задачи запуска с долгим
Debug.Log("Inventory manager starting..."); ¬ временем выполнения.
status = ManagerStatus.Started; ¬ Для задач с долгим временем выполнения используем состояние 'Initializing’.
}
}
Листинг 8.12. PlayerManager
using UnityEngine;
using System.Collections;
using System.Collections.Generic;
public class PlayerManager : MonoBehaviour, IGameManager { ¬ public ManagerStatus status {get; private set;}
public int health {get; private set;} public int maxHealth {get; private set;}
public void Startup() {
Debug.Log("Player manager starting...");
Наследуем класс и реализуем интерфейс.
health = 50; |
│ |
Эти значения могут быть инициализированы |
maxHealth = 100; │ |
сохраненными данными. |
status = ManagerStatus.Started;
}
8.3. Управление инвентаризационными данными и состоянием игры 203
public void |
ChangeHealth(int value) { ¬ Другие сценарии не могут напрямую задавать переменную health, |
|
health += |
value; |
но могут вызывать эту функцию. |
if (health |
> maxHealth) { |
|
health = |
maxHealth; |
|
} else if |
(health < 0) { |
|
health = |
0; |
|
} |
|
|
Debug.Log("Health: " + health + "/" + maxHealth);
}
}
Пока что InventoryManager — это оболочка, заполнением которой мы займемся позднее, в то время как PlayerManager уже обладает всей необходимой для нашего проекта функциональностью. Оба диспетчера наследуют от класса MonoBehaviour и реализуют интерфейс IGameManager. Это означает, что оба диспетчера получили всю функциональность MonoBehaviour, но при этом обязаны реализовывать структуру, заданную интерфейсом IGameManager. Эта структура состоит из одного свойства и одного метода, которые и определяют наши диспетчеры.
Свойство status доступно для чтения из любого места (функция чтения общедоступна), но задается только внутри нашего сценария (функция записи закрыта). Кроме того, оба диспетчера определяют метод Startup(). Их инициализация завершается сразу же (InventoryManager пока ничего не делает, в то время как PlayerManager задает пару значений), поэтому состоянию присваивается значение Started. Но частью инициализации модулей данных могут оказаться длительные задания (например, загрузка сохраненных данных), тогда загрузкой этих заданий займется метод Startup(), а состоянию диспетчера будет присвоено значение Initializing. После завершения заданий измените состояние на Started.
Замечательно — мы наконец готовы связать все вместе внутри одного главного диспетчера! Создайте сценарий Managers и введите в него код следующего листинга.
Листинг 8.13. Диспетчер диспетчеров!
using UnityEngine;
using System.Collections;
using System.Collections.Generic;
[RequireComponent(typeof(PlayerManager))] ¬ Гарантируем существование различных диспетчеров.
[RequireComponent(typeof(InventoryManager))]
public class Managers : MonoBehaviour {
public static PlayerManager Player {get; private set;} ¬ public static InventoryManager Inventory {get; private set;}
private List<IGameManager> _startSequence; ¬ Список диспетчеров, который просматривается в цикле во время стартовой последовательности.
void Awake() {
Player = GetComponent<PlayerManager>(); Inventory = GetComponent<InventoryManager>();
_startSequence = new List<IGameManager>(); _startSequence.Add(Player); _startSequence.Add(Inventory);
204 Глава 8. Добавление в игру интерактивных устройств и элементов
StartCoroutine(StartupManagers()); ¬ Асинхронно загружаем стартовую последовательность.
}
private IEnumerator StartupManagers() {
foreach (IGameManager manager in _startSequence) { manager.Startup();
}
yield return null;
int numModules = _startSequence.Count; int numReady = 0;
while (numReady < numModules) { ¬ Продолжаем цикл, пока не начнут работать все диспетчеры. int lastReady = numReady;
numReady = 0;
foreach (IGameManager manager in _startSequence) { if (manager.status == ManagerStatus.Started) {
numReady++;
}
}
if (numReady > lastReady)
Debug.Log("Progress: " + numReady + "/" + numModules); yield return null; ¬ Остановка на один кадр перед следующей проверкой.
}
Debug.Log("All managers started up");
}
}
Наиболее важной частью этого паттерна являются статические свойства на самом верху. Именно они позволяют другим сценариям использовать для доступа к различным модулям такой синтаксис, как Managers.Player или Managers.Inventory. Изначально эти свойства пусты, но они заполняются сразу же после запуска кода в методе
Awake().
СОВЕТ Подобно методам Start() и Update(), метод Awake() автоматически предоставляется классом MonoBehaviour. Как и метод Start(), он однократно запускается в начале выполнения кода. Но в принятой в Unity последовательности выполнения кода метод Awake() располагается перед методом Start(), отвечая за связанные с инициализацией задания, которые должны запускаться перед любыми другими модулями кода.
Кроме того, метод Awake() выводит последовательность запуска, а затем загружает сопрограмму, начинающую работу всех диспетчеров. А именно он создает объект List и прибегает к методу List.Add() для добавления всех диспетчеров.
ОПРЕДЕЛЕНИЕ Объект List представляет собой такую структуру данных, как коллекция из языка C#. Списки аналогичны массивам: в них хранятся последовательные наборы элементов определенного типа. Но размер списка может быть изменен в процессе работы, в то время как у массивов этот параметр не редактируется.