- Основное
- Создание Pull Request
- Описания коммитов
- Continuous Integration
- Процесс прохождения Code Review
- Договоренности по написанию кода
- Разработка безопасного кода
- Файлы
- Фишки и лайфхаки Dream Maker
Процесс работы с иссуями - гайд по иссуям и плашкам в нашем репозитории.
- Официальные языки нашего сообщества - русский и английский. Коммиты, пулл реквесты, иссуи - все можно писать на обоих языках.
- Мы ожидаем, что вы поможете нам поддерживать код, который вы добавили. Скорее всего мы к вам обратимся в случае возникновения каких-то проблем, связанных с вашими изменениями, в том числе рантаймами или багами.
Здесь приведен небольшой чек-лист важных вещей, на которые обязательно нужно обращать внимание при открытии Pull Request-а. Пренебрежение этим разделом может увеличить время проверки ваших изменений и принятия в сборку вплоть до бесконечности!
- Заголовок ПРа - цель его изменений. У любого Pull Request должна быть конкретная цель, которая должна быть указана в заголовке. Pull Request не должен содержать изменений, которые не относятся к указанной цели. Если цель Pull Request-а слишком большая, чтобы уместить её в одном заголовке - подумайте над тем, чтобы разбить свой Pull Request на более мелкие части.
- Изменения востребованы. Перед началом разработки убедитесь, что то, что вы хотите сделать - востребовано и будет принято. Для этого нужно либо обсудить изменения с текущими разработчиками сервера или геймдизайнером. Также изменения очевидно востребованы, если на них есть соответствующий иссуй, проверенный командой сервера. Чтобы ваши изменения ТОЧНО были приняты - лучше убедиться в том, что у вас есть соответстующий иссуй с подтверждением, так как любые договоренности в дискорде могут потеряться или оказаться неактуальными.
- Подробное описание. В описании Pull Request-а должно быть подробное описание того, зачем он нужен и что в нем происходит. Pull Request не должен содержать изменений, о которых ничего не написано в описании.
- Привяжите иссуи. Если Pull Request решает какие-то иссуи, то они должны быть указаны в описании специальным образом, например,
close #23
, чтобы Github смог их автоматически привязать к вашим изменениям. Подробнее можно почитать тут: https://help.github.com/articles/closing-issues-via-commit-messages. - Проверьте свой код. Открывайте Pull Request только если вы уверены в его работоспособности. Это означает, что вы проверили каждую строчку своих изменений, а также проверили свои изменения локально, запустив сервер на своем компьютере. Если что-то невозможно проверить полностью или вы считаете, что на какую-то часть изменений нужно обратить пристальное внимание проверяющих, то укажите это отдельно в описании. Конечно мы также будем проверять ваши изменения, но вы не должны полагаться на то, что ваши баги заметит кто-то другой и исправит за вас.
- Ваш код соответствует принятым договоренностям по написанию кода. Подробнее смотрите раздел "Договоренности по написанию кода" ниже.
Главная цель правильного именования и описания коммитов - упростить процесс чтения истории git. Чем понятнее написаны комментарии к коммитам, тем проще находить причины багов.
Коммиты следует именовать по следующей схеме:
action(area): short one-line description
long multiline description
related items
- action - тип действия, которое совершается в коммите. Например, fix, feat, refact, tweak, delete, docs и т.д. Отвечает на вопрос "Что делаем?"
- area - область кода, в которой происходит действие. Например, atmospherics, MC, changeling, exodus и т.д.
- related items - иссуи, которых касается коммит. Еще какие-то ссылки, которые могут относиться к PR-у. В случае иссуев обязательно использование ключевых слов Github для автоматического связывания коммита с иссуями.
Пример хорошего описания коммита:
tweak(vortex): more server-friendly vortex behavior and other tweaks
Rebalance.
Also fixes issue with O(N) intended malfunctions for 'battle function' becoming O(N^unknown) due to malfunction calling itself on each iteration.
Therefore, base malfunction chance was raised a bit.
Charge costs multiplied, as there was no 'bluespace cell' with 10k total capacity when previous values were implemented.
PR #1766
close #1432
Еще раз отдельно обратим внимание на то, что коммиты можно писать на русском языке по той же схеме. Однако area
и разные ключевые слова обязательно нужно писать на английском языке, согласно тому, как оно пишется в коде, чтобы всем было понятно какой части кода касаются ваши изменения.
Конфигурация загружается из нескольких источников в следующем порядке:
- Файлы в папке
config/default
- Файлы в папке
config/[MODE]
- вместоMODE
подставляется значение соответствующей переменной среды окружения. При запуске через VSCode автоматические устанавливается зачениеdev
. - Файлы в папке
config/[SERVER_ID]
- работает аналогично второму пункту. При запуске через VSCode используется занчениеtest
. Здесь можно хранить файлы конфигурации на время разработки и тестирования, так как они не учитываются гитом. - Переменные среды, формат записи:
ONYXBAY__KEY__VALUE
.
Caution
Все ключи на выходе приводятся к нижнему регистру.
Поддерживаемые форматы файлов конфигурации:
- JSON
- YAML
- INI
- JSON5
- TOML
Note
Файлы можно вкладывать в другие папки.
Каждый Pull Request проходит проверку на несколько вещей:
- Наличие и корректность чейнджлога.
- Наличие ошибок найденные статическим анализатором DreamChecker.
- Наличие нарушении стиля кода.
Наличие и корректность чейнджлога проверяется при каждом открытии Pull Request'а, редактировании текста в теле Pull Request'а, добавлении и удаления меток на Pull Request.
Остальные ошибки проверяются на каждый новый коммит один раз. Если возникает необходимость пропустить эти проверки - необходимо в сообщении коммита написать [ci skip]
.
- Аппрув разработчиков. Чтобы Pull Request был принят - он должен быть аппрувнут двумя активными разработчиками сервера.
- Аппрув OnyxBay owner-а. Если в Pull Request есть изменения игровой логики, то он должен быть аппрувнут OnyxBay Owner-ом. Аппрув не нужен, если Pull Request выполнен в рамках иссуя, который был ранее аппрувнут OnyxBay Owner-ом. Также аппрув не нужен для фикса багов и любых изменений, которые не затрагивают игровой опыт игроков (рефакторинг, оптимизации, фичи для администраторов и подобное).
- Проверка на локальной сборке. Принимаются только Pull Request-ы, проверенные локально. Если сам разработчик не проверяет свои изменения, то проверка проходит тщательнее, из-за чего принятие изменений затягивается.
- Вливание изменений в основную ветку. Если коммиты в PR соответствуют правилам именования, при мерже не возникает конфликтов и каждый коммит по отдельности не ломает сборку, то принимать следует с помощью rebase. Если коммиты в PR соответствуют правилам именования, но принять её с помощью rebase невозможно т.к. на каких-то коммитах из PR сборка сломана, или же из-за конфликтов, то принимать нужно с помощью merge. Если хотя бы один коммит в PR не соответствует правилам именования, то весь PR может быть принят только через squash с названием, соответствующим правилам.
- Вливание изменений на сервер. Ветка разработки называется
dev
. Перед обновлением серверов проверенные изменения изdev
вливаются вmaster
. В обоих ветках должна поддерживаться чистая история коммитов, которые соответствуют правилам именования. Сервера берут свой код из собственных веток, куда код вливается уже изmaster
. В собственных ветках серверов история может портиться засчет ревертов, мержей и временных изменений (например, для ивентов). В связи с техническими особенностями хоста ветки серверов нельзя обновлять через force-push. На сервер изменения должны попадать вместе с чейнжлогом для максимального оповещения игроков о статусе разработки.
Имя переменной, функции или типа должно сообщить, почему эта переменная существует, что она делает и как используется. Если имя требует дополнительных комментариев, значит, оно не передает намерений программиста. Лучше написать, что именно измеряется и в каких именно единицах.
Не бойтесь использовать длинные имена: длинное содержательное имя лучше короткого невразумительного.
Пример хорошего названия переменной: days_since_creation; Цель: убрать неочевидность.
Неинформативные слова избыточны. Слово variable никогда не должно встречаться в именах переменных. Слово object никогда не должно встречаться в именах объектов. Чем имя name_string лучше name? Разве имя может быть, скажем, вещественным числом?
russian_to_utf намного лучше rustoutf.
Чем дольше время жизни переменной - тем длиннее должно быть её название. Однобуквенные имена могут использоваться только для локальных переменных в коротких методах.
Имена типов должны представлять собой существительные и их комбинации: traitor, profession, id_card и request_parser. Старайтесь не использовать в именах классов такие слова, как manager, processor, data или info. Имя класса не должно быть глаголом.
Имена методов представляют собой глаголы или глагольные словосочетания: explode_station, delete_ian, save и т. д. Методы чтения/записи образуются из значения и префикса get, set и is.
DM позволяет писать код блоками:
datum
datum1
var
varname1 = 1
varname2
static
varname3
varname4
proc
proc1()
code
proc2()
code
datum2
varname1 = 0
proc
proc3()
code
proc2()
..()
code
Такой способ написания делает невозможным текстовый поиск функции или типа по коду. Единственное исключение - внутри блока описания типа можно определять переменные.
Тот же код, написанный правильно:
/datum/datum1
var/varname1
var/varname2
var/static/varname3
var/static/varname4
/datum/datum1/proc/proc1()
code
/datum/datum1/proc/proc2()
code
/datum/datum1/datum2
varname1 = 0
/datum/datum1/datum2/proc/proc3()
code
/datum/datum1/datum2/proc2()
..()
code
- Первое правило: функции должны быть компактными.
- Второе правило: функции должны быть еще компактнее.
Серьезные программисты работающие в реальном мире ценой многих проб и ошибок вывели, что функции должны быть очень маленькими. Желательно, чтобы длина фукнции не превышала 20 строк. В условиях разработки SS13 кому-то это может показаться излишним, но даже в наших проектах можно заметить, что чем длиннее функция тем сложнее её поддерживать, тем больше в ней багов и тем меньшее количество людей хочет в неё лезть и с ней разбираться.
Функция должна выполнять только одну операцию. Она должна выполнять ее хорошо. И ничего другого она делать не должна. Если функция выполняет только те действия, которые находятся на одном уровне под объявленным именем функции, то эта функция выполняет одну операцию.
Функцию, выполняющую только одну операцию, невозможно осмысленно разделить на секции.
Функция должна что-то делать или отвечать на какой-то вопрос, но не одновременно. Либо функция изменяет состояние объекта, либо возвращает информацию об этом объекте. Совмещение двух операций часто создает путаницу.
Закомметированный код запрещен.
Запрещены комментарии отвечающие на вопрос "Что делает этот код?". Вместо этого используете методы для увеличения читаемости кода выше, чтобы код объяснял себя сам. Разрешены комментарии отвечающие на вопрос "Зачем этот код нужен?".
Также можно писать комментарии типа "TODO" и "FIXME".
Запрещено использовать оператор :
. Всегда явно преобразуйте переменную к конкретному типу.
Плохой пример:
var/something_general_object = ...
something_general_object:specific_type_func()
Мы узнаем, что у something_general_object нет такой функции только когда запустим сервер. Более того, когда мы это узнаем - будет непонятным что это за тип, почему у него было этой функции и какой тип на самом деле ожидался. С другой стороны, в коде:
var/something_general_object
var/more/specified/type/O = object
ASSERT(istype(O)) // bad argument
O.specific_type_funс()
Мы явно указываем какой тип мы ожидаем и статически проверяем, что у этого типа есть такая функция.
Если ваша функция расчитана на то, чтобы работать только с аргументами определенного типа или какими-то конкретными значениями - используйте ASSERT, чтобы явно это указать. Если, вдруг, случится так, что откуда-то в функцию приедет неожиданный аргумент - лучше сразу зарепортить об этом рантаймом и выйти из функции, чем пойти дальше, сделать что-то неожиданное и, может быть, даже не оставить следа о том, что фукнция сработала не так, как нужно.
Имейте в виду, что ASSERT генерирует рантайм, что ведет к выходу из функции с возвратом null. Функция выше по стеку продолжит свое выполнение с полученным null.
Плохой пример:
/datum/controller/subsystem/open_space/proc/add_turf(turf/T, ...)
if(!isturf(T))
return
...
Если мы в функцию add_turf
передали аргументом не turf - это очевидный баг. С таким вариантом реализации этот баг останется незамеченным и он может привести к неожиданным проблемам. Но если мы напишем так:
/datum/controller/subsystem/open_space/proc/add_turf(turf/T, ...)
ASSERT(isturf(T))
...
То мы все также выйдем из функции, вернув null, но при этом запишем рантайм, который мы увидим в логах, благодаря чему сможем обнаружить и пофиксить изначальный баг, который приводит к тому, что сюда аргументом передается неверный тип.
Например: /datum/thing
, вместо datum/thing
Например: /datum/thing/blue
, вместо datum/thing/BLUE
или datum/thing/Blue
Локальные переменные всегда определяйте в формате var/type/name
, вместо var type/name
. В аргументах функций оно избыточно, поэтому всегда опускаем и пишем просто type/name
.
Для индентации (отступ до текущего блока кода) всегда используем только табы. После индентации может быть выравнивание пробелами (не выравнивайте табами - иначе разъедется в редакторах с разной длиной таба).
Когда вы копируете один и тот же код в разные места - появляется необходимость одинаково поддерживать этот код в двух местах. При любом изменении оригинального кода надо помнить о копии этого кода в другом месте и зачастую вносить изменения и туда.
Чтобы избежать подобных проблем используйте наследование объектов друг от друга или же просто вынесите необходимую логику в отдельную функцию.
Контроллеры, используемые в нашей сборке, должны справляться с длительными операциями и лагами, но они не могут контролировать то, что происходит во время загрузки карты, когда для всех объектов вызывается New. Для любых новых объектов, без явной необходимости, используйте Initialize
, чтобы сделать все, что вы хотели сделать в New
. Это уменьшает количество функций вызываемых на этапе загрузки карты. Чтобы узнать больше про то, как работает Initialize
, смотрите https://github.com/ChaoticOnyx/OnyxBay/blob/dev/code/game/atoms.dm
NB: Важно понимать то, как работают Initialize и последовательность их выполнения относительно New, иначе можно словить приколы (см. #3817). Initialize вызывается в /atom/New. Соответственно при создании объекта сначала вызывается New последнего типа в иерархии наследования. Дальше через ..() (если они, конечно, есть) последовательно вызываются New более ранних по иерархии типов вплоть до /atom/New, где вызывается /atom/Initialize, который точно также начинает выполняться с последнего переопределения в иерархии наследования объектов.
Таким образом получается, что в любом New, все что находится после ..() - вызывается после всех Initialize в цепочке наследования, а все что написано до ..() - работает перед всеми Initialize в цепочке наследования. Из этого следует, что если в вашей иерархии типов перемешиваются New и Initialize, то порядок их вызова может быть немного непредсказуемым, что вызывает приколы вроде перезаписи переменной, которая определяется в чайлдовом типе, в Initialize, значением из родителького объекта, из New. Пример и подробный разбор такой баги можно посмотреть здесь: #3817 .
Если вы не очень поняли что написано выше, то просто следуйте одному простому правилу:
Старайтесь не перемешивать New и Initialize в иерархиях объектов. Если меняете один New на Initialize - меняйте сразу всей иерархии. Если в иерархии используются New, то проще использовать New в новом объекте этой иерархии, однако в плане производительности - лучше переписать всю иерархию на Initialize.
Все объекты должны всегда помечаться к удалению с помощью qdel
.
Использование del
запрещено, так как эта функция запускает довольно требовательную процедуру поиска и обнуления ссылок на объект по всему коду.
Просто "бросать" объекты без вызова qdel
не рекомендуется, так как в таком случае не будет вызван Destroy, который, даже если еще не реализован для конкретного объекта, кто-то может захотеть добавить позже.
Отдельно важно помнить про случаи с циклическими зависимостями, когда объекты хранят ссылки друг на друга циклически. Если такие объекты просто пометить к удалению через qdel
, то объекты никогда не будут удалены, так как сборщик мусора будет ждать, пока ссылки на объекты закончатся (а наши объекты циклически указывают друг на друга - поэтому ссылки никогда не закончатся). В таком случае нужно делать не только помечать объекты к удалению через qdel
, но и занулять какую-то часть ссылок так, чтобы избавиться от цикла. С помощью макроса QDEL_NULL(obj)
можно одной строчкой пометить obj
к удалению и занулить ссылку.
Пример реализации циклических объектов:
/mob/living/simple_animal
...
var/datum/mob_ai/mob_ai
/mob/living/simple_animal/Initialize()
. = ..()
mob_ai = new() // simple_mob создает и владеет своим искусственным интеллектом
mob_ai.holder = src // искусственному интеллекту нужно знать кем он управляет
// получили циклическую ссылку simple_mob <-> mob_ai объектов друг на друга
/mob/living/simple_animal/Destroy()
QDEL_NULL(mob_ai) // когда моба удаляют с карты нам нужно разорвать циклическую ссылку. Убиваем ссылку на mob_ai.
// в итоге на mob_ai больше никто в мире не ссылается, сборщик мусора его собирает, а следом собирает simple_mob.
// the same as:
// qdel(mob_ai) // тоже самое, что делает макрос QDEL_NULL в раскрытом виде
// mob_ai = null
return ..()
В отличие от loc = newpos
, forceMove
может переопределяться и в нем написано много важной логики, которая нужна при перемещении объектов из одного места в другое.
В нашем репозитории был принят стандарт, что вместо проверки stat
на DEAD
вызывается специальные функции для разных сценариях.
"Игровая" смерть - персонаж не мертв, но внутреигровые приборы будут отображать, что он умер - is_ic_dead
Пример использования смотри в коде медицинского сканера.
Фактическая смерть - используется в тех случаях, когда персонаж 100 процентов не может быть возвращен в жизни - is_ooc_dead
Пример использования смотри в коде связанные с выходом в призрака.
В коде не должно быть "брошенных" числовых/строчных или каких-либо еще значений, смысл которых неясен из контекста или которые могут быть переиспользованы. Определяйте их через макрос, если они нужны в глобальном скоупе, или создавайте локальную переменную с понятным названием.
Пример:
/datum/proc/do_the_thing(thing_to_do)
switch(thing_to_do)
if(1)
(...)
if(2)
(...)
Здесь неясно, что означают "1" и "2"! Вместо этого можно было бы написать:
#define DO_THE_THING_REALLY_HARD 1
#define DO_THE_THING_EFFICIENTLY 2
/datum/proc/do_the_thing(thing_to_do)
switch(thing_to_do)
if(DO_THE_THING_REALLY_HARD)
(...)
if(DO_THE_THING_EFFICIENTLY)
(...)
Так получается гораздо понятнее, что увеличивает читаемость вашего кода.
Еще пример:
/datun/gamemode/proc/create_events()
events.Add(new /datum/event(90, list(/datum/role/hos)))
events.Add(new /datum/event(50, list(/datum/role/officer)))
В примере понятно, что мы задаем ивенты для какого-то игрового режима, однако непонятно что за аргументы передаются внутрь - это как раз "магические" значения. Чтобы сделать понятнее, можно, например, явно указать переменные, которые мы присваиваем:
/datun/gamemode/proc/create_events()
events.Add(new /datum/event(chance=90, requires_roles=list(/datum/role/hos)))
events.Add(new /datum/event(chance=50, requires_roles=list(/datum/role/officer)))
(if, while, for и другие)
- Все операторы контроля выполнения не должны содержать другого кода на той же строчке (
if (blah) return
) - Все операторы контроля выполнения, сравнивающие переменную с каким-то значением, должны использовать формулу
переменная
оператор
значение
, не наоборот (например:if (count <= 10)
, вместоif (10 >= count)
)
Оператор in имеет наименьший приоритет (это не указано в рефе), поэтому его всегда нужно заключать в скобки: if(A && (A in foo_list))
.
Потому что не имеет смысла
Вместо to_chat(usr, "<span class="warning">[text]</span>")
используйте to_chat(usr, SPAN_WARNING(text))
.
Не стоит строчить многоуровневые конструкции из блоков if, когда того же результата можно достичь ранним возвратом из функции
Вот так делать не стоит:
/datum/datum1/proc/proc1()
if (thing1)
do stuff
if (!thing2)
do more stuff
if (thing3 == 30)
do extra stuff
А так уже лучше:
/datum/datum1/proc/proc1()
if (!thing1)
return
do stuff
if (thing2)
return
do more stuff
if (thing3 != 30)
return
do extra stuff
Во всех случаях, когда нужно логическое значение, вместо 1/0 используйте TRUE/FALSE.
В DM есть ключевое слово global, которое используется в двух случаях:
- Внутри типа (или внутри функции) мы хотим определить переменную общую для всех объектов этого типа (для всех вызовов этой функции). (http://www.byond.com/docs/ref/#/var/global).
- Внутри функции мы хотим использовать переменную из глобального скоупа, когда у нас в локальном скоупе уже есть переменная с таким же названием. (http://www.byond.com/docs/ref/#/proc/var/global) Так использовать global не стоит! См. следующее правило.
Так как в первом случае речь идет не о видимости переменной, а том, что она общая для разных экземпляров типа (вызовов функции), то название ключевого слова global может кого-то запутать. Поэтому вместо него в первом случае мы стараемся использовать ключевое слово static, которое отсутствует в документации, но полностью заменяет global в этом случае и лучше описывает суть происходящего.
Отдельно обратите внимание, что все переменные в BYOND имеют глобальную видимость с точки зрения доступа, поэтому global (и static) в глобальном скоупе не имеет смысла и запрещено.
Главная проблема глобальных переменных в том, что чем больше они используются, тем сложнее понять логику их изменения и поведения процессов, которые от них зависят. Поэтому глобальные переменные лучше вовсе не использовать. Однако, если вам все-таки очень нужна глобальная переменная, то, как минимум, создавайте их на контроллере глобальных переменных, что даст чуть больше возможностей для их дебага:
Там все просто - вместо глобальных перменных нужно всегда использовать макросы (ну и конечно же лучше всего вообще не использовать глобальные переменные). Пример с тг:
Вместо:
var/X
var/list/Y
var/datum/genitalia/Z
var/A = 42
var/list/B = list(burn = "witch")
var/datum/genitalia/C = MakeAPenis()
var/hub_password
Используйте:
GLOBAL_VAR(X)
GLOBAL_LIST(Y)
GLOBAL_DATUM(Z, /datum/genitalia)
GLOBAL_VAR_INIT(A, 42)
GLOBAL_LIST_INIT(B, list(burn = "witch"))
GLOBAL_DATUM_INIT(C, /datum/genitalia, MakeAPenis())
GLOBAL_PROTECT(hub_password)
range()
, view()
, hearers()
, locate() [in world]
, for(... [in world])
и подобные операции очень требовательные в плане производительности, поэтому должны импользоваться как можно реже. Зачастую их можно заменить глобальным списком объектов нужного типа.
alert()
и input()
ждут реакции пользователя и пока они ждут, все окружение может поменяться: поля объекта, переменные вокруг и т.д. Учитывайте это и добавляйте дополнительные проверки по необходимости.
Практика показывает, что addtimer
лучше оптимизирован, чем встроенные spawn и sleep, плюс код, написанный с его использованием, гораздо проще дебажить.
- Всегда относитель к пользовательскому вводу так, как будто он намеренно пытается все сломать. Проверяйте пользовательский ввод на все случаи, которые не соответсвуют ожиданиям вашего кода. Для чисел проверяйте границы, для строк используйте escape-функции. Обратите внимание на функцию
sanitize
, которой удобно эскейпить вообще любойinput
, чтобы избежать ввод какого-то кода, который выполнится в браузере. - Обязательно эскейптьте все команды к базе данных - используйте
sql_query
чтобы обработать весь текст от игроков и админов перед тем как передавать его в базу данных. Для чисел используйтеisnum
. - Все вызовы топиков обязательно нужно проверять на их корректность. Такие вызовы могут быть подделаны со стороны клиента, поэтому их содержимым может быть что угодно!
- Скрывайте от игроков любую информацию, которая может быть использована для метагейма (даже такую простую, как количество игроков, которые нажали Declare, так как даже она может быть использована, чтобы вычислить текущий режим).
- Когда вы пишите код, который может каким-то образом влиять на раунд и генерировать ВЕСЕЛЬЕ, дважды проверьте, что такой функционал будет доступен только админам соответствующего ранга.
- Не используйте
locate()
для глобального поиска экземпляра типа который может быть удалён в процессе игры. Эта функция может вернуть экземпляр который находится в процессе удаления но не до конца удалён. Вместо этого лучше держать список с ещё не удаленными объектами и искать в нём (locate() in objects_list
).
- Рантаймы не содержат полного пути до файла - поэтому избегайте одинаковых названий файлов даже в разных папках.
- Названия файлов не должны содержать пробелов или символов, которые придется эскейпить, указывая uri.
- Названия файлов и все пути всегда должны быть в нижнем регистре, чтобы избежать проблем, связанных с разным отношением к регистру в разных операционных системах.
Как и любые другие языки, в BYOND есть свои особенности, которые стоит учитывать, чтобы писать более эффективный код. Тут описаны некоторые из них.
for(var/i = 1, i <= some_value, i++)
является стандартным способом писать циклы во многих языках программирования, однако в BYOND, внезапно, for(var/i in 1 to some_value)
оказывается быстрее в плане производительности. (Обратите внимание, что to
включает и левую, и правую границу).
ОДНАКО, если some_value
или i
меняются в течение цикла, или вы итерируете по элементам списка, длина которого изменяется, вы НЕ можете использовать этот тип цикла for.
Оператор (A || B) возвращает выражение A или B без изменений их значений. Этот факт можно использовать, чтобы кратким образом подставить B вместо нулевого (или ложного) A.
Пример:
obj.name = some_name || "Unknown"
Этот код полностью аналогичен следующему:
if (some_name):
obj.name = some_name
else
obj.name = "Unknown"
Если some_name - это имеет ложное значение, то obj.name будет присвоено значение "Unknown".
Начиная с версии 514 в BYOND добавили сокращенную версию оператора: A ||= B.
Аналогичным образом работает и оператор &&, благодаря чему можно писать условия в одну строчку. Такое сокращение не настолько полезно, как предыдущее, но тоже может использоваться в разных ситуациях.
Пример:
var/obj/a = some_object
istype(a) && a.foo()
Этот пример равнозначен коду:
var/obj/a = some_object
if(istype(a))
a.foo()
Начиная с версии 514 в BYOND добавили сокращенную версию оператора: A &&= B.
По дефолту BYOND проверяет в рантайме тип элемента при итерации по списку:
for(var/obj/item/projectile/O in weapon.content)
В этом примере из списка content
будут отобраны только объекты типа /obj/item/projectile
(объекты других типов будут пропущены), но ради такого поведения BYOND будет неявно проверять тип в каждой итерации, что влияет на перфоманс. Обычно это не очень критично, но если цикл вызывается часто или в нем много элементов, то такие циклы есть смысл оптимизировать.
as anything() позволяет избавиться от неявной проверки типа:
for(var/obj/item/projectile/O as anything() in weapon.content)
Теперь проверки на тип не будет и обрабатываться будут все объекты из списка. Если случайно в списке окажется объект не являющийся /obj/item/projectile
, то если вы попытаетесь вызывать методы /obj/item/projectile
у такого объекта, то вы получите runtime ошибку. Таким образом следить за типом объектов в списке нужно самостоятельно, но цикл будет работать быстрее.
Заметим, что следующие циклы равнозначны по своей логике работы:
for(var/obj/item/projectile/O in weapon.content)
...
for(var/obj/item/projectile/O as anything() in weapon.content)
if(!istype(O)) // "лишняя" проверка, от которой можно избавиться,
// если мы знаем, что в weapon.content могут храниться только /obj/item/projectile
continue
...
Также вместо as anything() можно писать просто as().
В случае добавления новых стен (либо иных объектов, подразумевающих использование автотайлинга) настоятельно рекомендуется обратить внимание на функцию BakeBitmaskOverlays()
у подсистемы SSmisc. Зто позволит сократить количество одновременно отображаемых на экране оверлеев, и как следствие снижает нагрузку на игровой клиент.
Однако, не стоит забывать, что это касается только объектов, не использующих отрисовку диагональных соединений (например, столы и окна), т.к. для них потребуется запекание не 16, а 256 вариантов, что может привести (и обязательно приведёт) к огромной задержке во время инициализации.