Функции¶
Ü поддерживает свободные функции. Функции принимают от нуля и более аргументов и могут возвращать значение.
Примеры объявлений функций:
fn Add( i32 x, i32 y ) : i32; // Функция принимает два аргумента типов "i32" и возвращает значение типа "i32".
fn Sleep( f32 sec ); // Функция принимает один аргумент типа "f32" и не возвращает значения.
fn GetPi() : f64; // Функция не принимает аргументов и возвращает значение типа "f64".
fn Max( i64& x, i64& y ) : i64&; // Функция принимает два аргумента типов "i64" по ссылке и возвращает ссылку того же типа.
Аргументы¶
Функции могут принимать аргументы-значения и ссылочные аргументы.
Ссылочные аргументы объявляются с использованием &
после имени типа аргумента.
Аргументы могут иметь модификаторы изменяемости mut
или imut
, указываемые после имени типа или после &
(для ссылочных аргументов).
Если модификатор изменяемости не указан, неявно будет применён модификатор imut
.
Для ссылочных аргументов модификаторы изменяемости имеют значения для вызываемого кода. В функцию, принимающую изменяемую ссылку, нельзя передать неизменяемую переменную. Для аргументов значений же модификатор изменяемости влияет только на тело функции, неизменяемый аргумент-значение нельзя будет изменить в теле.
Примеры аргументов:
fn Add( i32 imut x, i32 y ) : i32; // Неизменяемые аргументы-значения. Второй аргумент будет неизменяемым, т. к. к нему применён модификатор по умолчанию "imut"
fn Inc( u32 &mut x ); // Изменяемый ссылочный аргумент
fn GetLength( ust::string8 &imut str ) : size_type; // Неизменяемый ссылочный аргумент
fn GetNumZeros( ust::string8 & str ) : size_type; // Неизменяемый ссылочный аргумент, т. к. к нему применён модификатор по умолчанию "imut"
Возвращаемое значение¶
Функция может не возвращать значений. Для этого можно или не указать тип возвращаемого значения, или указать тип void
:
fn Sleep( f32 sec );
fn USleep( u64 ns ) : void;
Функция может возвращать значение. Для этого надо указать тип значения:
fn Add( i32 x, i32 y ) : i32; // тип возвращаемого значения - "i32"
Функция может возвращать ссылочное значение - изменяемое или неизменяемое. Для этого надо указать &
после типа возвращаемого значения.
После него можно опционально указать модификатор изменяемости mut
или imut
.
Если модификатор изменяемости не указан, неявно будет применён модификатор imut
.
fn Max( i64& x, i64& y ) : i64 &imut; // Функция вернёт неизменяемую ссылку типа "i64"
fn Max( u64& x, u64& y ) : u64 &; // Функция вернёт неизменяемую ссылку типа "u64", модификатор "imut" применён неявно
fn Max( u64 &mut x, u64 &mut y ) : i64 &mut; // Функция вернёт изменяемую ссылку типа "u64"
Перегрузка функций¶
Возможно объявить более одной функции с одним и тем же именем в одном пространстве имён. Но у всех одноимённых функций должны различаться параметры - числом, типом, изменяемостью (для ссылочных параметров).
fn Foo();
fn Foo(i32 x); // Ок, перегрузка с другим количеством параметров
fn Foo(f32 &imut x); // Ок, перегрузка с другим типом параметра
fn Foo(f32 &mut x); // Ок, перегрузка с с отличной изменяемостью параметра
Если у функций различается только что-то кроме параметров (возвращаемое значение, модификатор unsafe
), то перегрузка невозможна.
Также невозможна перегрузка с различием только в изменяемости параметров-значений или перегрузка неизменяемого параметра по модификатору ссылочности.
fn Foo() : i32;
fn Foo() unsafe : f32; // Ошибка, перегрузка невозможна - сигнатура функций одинакова
fn Bar(i32 mut x);
fn Bar(i32 imut x); // Ошибка, перегрузка невозможна - различается только изменяемость параметра-значения
fn Baz(i32 imut x);
fn Baz(i32 &imut x); // Ошибка, перегрузка невозможна - различается только ссылочный модификатор неизменяемого параметра
Прототипы и реализации¶
Объявление функции, оканчивающееся на ;
означает, что объявлен лишь прототип функции, если же после объявления функции следует блок её тела, обрамлённый в {}
, то это объявление реализации.
// объявили прототип
fn Add( i32 x, i32 y ) : i32;
// позже объявили реализацию функции
fn Add( i32 x, i32 y ) : i32
{
return x + y;
}
Вывод типа возвращаемого значения¶
Вместо типа возвращаемого значения функции можно указать ключевое слово auto
.
В этом случае тип возвращаемого значения выведется автоматически из оператора return
.
fn Div( i32 x, i32 y ) : auto
{
return x / y; // Тип будет "i32"
}
fn Abs( f32 x ) : auto
{
// Во всех операторах "return" тип - "f32"
if( x >= 0.0f ) { return x; }
return -x;
}
Функции с выводом типа возвращаемого значения имеют ряд ограничений:
Они должны иметь тело
Они не могут рекурсивно звать сами себя
Они не могут быть членами структур или классов
Условное наличие функции¶
Иногда необходимо, чтобы функция отсутствовала, в зависимости от какого-либо условия.
Особенно это может быть полезно в шаблонном коде. Для этого в Ü существует специальная конструкция - enable_if
.
Эта конструкция может быть указана после fn
, опциональных constexpr
, virtual
, nomangle
в объявлении функции.
После этого указывается выражение, в ()
скобках. Выражение должно быть константой времени компиляции и иметь тип bool
.
Если выражение ложно - функция построена не будет, дальнейший её заголовок и тело не будут обрабатываться.
auto constexpr is_32bit = typeinfo</size_type/>.size_of == 4s;
// Функция будет существовать только на 32-битной платформе
fn enable_if( is_32bit ) Bar();
fn Foo()
{
Bar(); // На 64-битной платформе будет ошибка компиляции - функция не найдена.
}