.. keyword:: function inline-closure definition: function ( ) : { } inline-closure literal: (: ; , ... , :) :topics: closures, keyword, functions :name: inline-closures :highlighters: function \(: :\) Inline closures are a way to program closures which are compiled at the time an object is loaded, but can access values from their enclosing function at runtime. .. note:: Inline closures are not to be confused with inline functions known from other languages. In the definition of an inline closure, some elements are optional:: defaults to 'mixed' ( ) defaults to '(mixed $1 ... mixed $9)' : no default The special ``(: :)`` form is meant for simple expressions (and MudOS compatibility). The form:: (: ; , ..., :) is the shorthand notation for:: function { ; return , ..., ; } For example the two statements:: sort_array(arr, function { return $1 < $2; } ) sort_array(arr, (: $1 < $2 :) ) do the same. The example also demonstrates that both the :part:`statements` and the :part:`expr` part in this form are optional. .. note:: :macro`__LPC_INLINE_CLOSURES__` is defined when the inline closures as described here are available. If not defined, the driver implements a more restricted version (``(: :)`` syntax only, no context variables) for backwards compatibility. .. usage:: The closure below 'inherits' the current value of the local variable 'arg' at the time the closure is created. These values, called the "context" of the closures, are stored in a special set of variables in the closure:: closure factory (int arg) { return function int (int val) { return val * arg; }; } closure f1 = factory(2); closure f2 = factory(3); funcall(f1, 3); // returns 6. funcall(f2, 3); // returns 9. The closure can permanently change its context:: closure factory (int arg) { return function int (int val) { return val * arg++; }; } closure f = factory(2); funcall(f, 3); // will yield 6. funcall(f, 3); // will now yield 9! .. note:: Changes to the closure context will not reflect on the local variable it was copied from or vice versa. In addition to the implicit context inherited from the defining function, additional context variables can be defined in the closure:: closure factory (int arg) { return function int (int val) : int x = 2 * arg { return val * x; }; } closure f = factory(2); funcall(f, 3) -> will yield 12. It is possible to define multiple context variables with and without initialisation:: closure factory (int arg) { return function int (int val) : int y, x = 2 * arg; int z { return val * x; }; } These explicit context variables are useful when the closures needs to keep a state, or to improve performance:: mapping m = ...; closure slow (int arg) { return function mixed () { return m[arg]; } } closure fast (int arg) { return function mixed () : mixed val = m[arg] { return val; } } In the above example, the ``fast()`` function executes the lookup m[arg] only once when the inline closure is created; the ``slow()`` function on the other hand returns a closures which looks up ``m[arg]`` every time it is called. A second effect is that the results of the slow closure change when ``m`` changes; the result of the fast closure is always the same. :history 3.2.7 changed: implemented the older, restricted form of inline closures. :history 3.3.271 changed: implemented the full form of inline closures. :history 3.3.275 changed: re-allowed statements in the ``(: :)`` form. .. seealso:: `closures-abstract(LPC)`, :topic:`closures-example`, :guide:`closures` :type:`closure`