Uniform Function Call Syntax

Uniform Function Call Syntax (UFCS) or sometimes Universal Function Call Syntax is a programming language feature in D and Rust[1] that allows any function to be called using the syntax for method calls (as in object-oriented programming), by using the receiver as the first parameter, and the given arguments as the remaining parameters.[2] UFCS is particularly useful when function calls are chained,[3] behaving similar to pipes, or various dedicated operators available in functional languages for passing values through an expression. It allows free-functions to fills a role similar to extension methods in some other languages. Another benefit of the method call syntax is use with 'dot-autocomplete' in IDEs, which use type information to show a list of available functions, dependant on the context. When the programmer starts with an argument, the set of potentially applicable functions is greatly narrowed down, aiding discoverability.

C++ proposal

It has been proposed (as of 2016) for addition to C++ by Bjarne Stroustrup[4] and Herb Sutter, to reduce the ambiguous decision between writing free functions and member functions, to simplify the writing of templated code. Many programmers are tempted to write member-functions to get the benefits of the member-function syntax, however this leads to excessive coupling between classes.

Examples

D programming language

import std.stdio;

int first(int[] arr)
{
    return arr[0];
}

int[] addone(int[] arr)
{
    int[] result;
    foreach (value; arr) {
        result ~= value + 1;
    }
    return result;
}

void main()
{
    auto a = [0, 1, 2, 3];

    // All the followings are correct and equivalent
    int b = first(a);
    int c = a.first();
    int d = a.first;

    // Chaining
    int[] e = a.addone().addone();
}

Rust programming language

trait Foo {
    fn f(&self);
    fn g(&self);
}

trait Bar {
    fn g(&self);
}

struct Qux;

impl Foo for Qux {
    fn f(&self) { println!("Qux’s implementation of Foo::f"); }
    fn g(&self) { println!("Qux’s implementation of Foo::g"); }
}

impl Bar for Qux {
    fn g(&self) { println!("Qux’s implementation of Bar::g"); }
}

fn main() {
    let q = Qux;

    // These two are equivalent:
    q.f();
    Foo::f(&q);

    // This would not work because .g() is ambiguous:
    // q.g();
    // But it's possible to disambiguate using UFCS
    Foo::g(&q);
    Bar::g(&q);
}

See also

References


This article is issued from Wikipedia - version of the 12/2/2016. The text is available under the Creative Commons Attribution/Share Alike but additional terms may apply for the media files.