Functions
Function Definition
Functions (or lambdas) are a type of binding which can accept a set of inputs and gives an output.
For example fn(int,int -> int)
defines a function type (which accepts two integer numbers and gives an integer number) and fn(x:int, y:int -> int) { x+y }
is a function literal which given two numbers, returns their sum.
Declaration
functionName = fn(name1: type1, name2: type2... -> OutputType1, OutputTyp2, ...) { code block }
Function names must be camelCased.
Functions contain a set of bindings and the last expression in the code block determines function output (except for errors which can be returned early using at operator).
If calling a function that returns multiple bindings, you can use
_
to ignore some of them.There is no function overloading. Functions should have unique names in their container module.
You can alias a function by defining another binding pointing to it (example A).
If a function has no input, you can can eliminate input/output type declaration part (Example B). In this case, compiler will infer output type.
If a function is being called with literals (compile time known values), compiler will try to evaluate it during compilation (e.g. generics).
Module level functions that start with
_test
and have no input/output are considered unit test functions. You can later instruct compiler to run them (Example D).There is
assert
core function that can be used for checking assertions. You can disable assertions with a compiler flag.You can chain multiple nested function calls in reverse order via
::
operator (Example E).If a function returns multiple outputs you can simply send its output to another function if they match (Example F).
Examples
Lambdas
All functions are lambdas.
Functions are closure. So they have access to bindings in parent contexts (Module or parent function).
You can use
_
to create a lambda based on an existing function. Just make a normal call and replace the lambda inputs with_
(Example A).If lambda is assigned to a variable, it can invoke itself from the inside (Example B). This can be used to implement recursive calls.
Examples
Optional Arguments
When defining a function, you can use :>
instead of :
between argument name and type to indicate that, if that argument is missing when function is called, compiler or runtime should pass a default value for that argument. This default value is determined based on the type of the argument.
If type is union with an option for
nothing
, then nothing is the default value.If type is a number (int, float, char, byte), zero is the default value.
There are other default values for generic types which are explained in Generics section.
Note that optional arguments should be placed at the end of argument list.
Function Call Resolution
We use a static dispatch for function calls. So the target function in a function call is determined at compile time.
Also because you cannot have two functions with the same name, it is easier to find what happens with a function call.
If
MyInt = int
, you cannot call a function which needs anint
with aMyInt
binding. Because as mentioned before, these two are completely different types.Function resolution is done similar to type name resolution.
Parameter types must be "identical, exchangeable or compatible" to function argument types, or else there will be a compiler error.
For example if function argument type is
int | nothing
and parameter is anint
it is a valid function call (But not the other way around).When you need a function of type
fn(T->U)
any function that can accept T (or more, likeT|Y
) and returns U (or less) works fine.
Last updated
Was this helpful?