Skip to content

Simple and surprisingly powerful currying implemented concisely in c++20

Notifications You must be signed in to change notification settings

EvanMcGorty/cpp-currying

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

18 Commits
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

C++ Currying

I whipped this up one morning after realizing I had never seen an implementation of actual currying in c++, let alone something ergonomic and powerful. I then continued working on it, fixing bugs and changing/optimizing the design.

The generic class "curry" (contained in curry.h) transforms a function, so that there is no difference between, for example, curry{f}(x,y,z), curry{f}(x)(y,z), curry{f}(x,y)(z), and curry{f}(x)(y)(z), regardless of if/how the original function was curried. The constructor can also take arguments to bind, so curry{f,x,y,z} can also be used.

The result of an application can then be retrieved via implicit conversion, or alternatively via the public field curry::curry_wrapped_val (which is of type curry::curry_wrapped_val_type, the same type as the template parameter).

Functions with no arguments must be explicitly invoked with empty operator(), as if they take a unit type. Note that curry{f,args...} is not equivalent to curry{f}(args...) when args is empty.

Intermediate function types of already partially-curried functions are not lost from being wrapped up into anonymous functions and can thus be retrieved via implicit conversion (so types with overloads of operator() won't just automatically get eaten up by curry).

The semantics are nearly identical to those of std::bind_front, so the user is expected to use std::ref/cref as necessary. The only exception is that curry::operator() called from a (lvalue) reference will wrap the callable object in a std::ref. It is worth noting that, as with std::bind front, this prevents it from later being able to have its rvalue operator() called, so explicitly copying and/or std::move-ing it to avoid this is often desireable. Also, due to the nature of bind_front, basically any number of arguments of any type can be applied to a curried value and it will compile until someone tries actually extracting the return type. This means that std::invokable will always be satisfied as it doesnt allow you to specify a return type (in contrast with std::is_invokable_r).

There is also a concept, "curried" (contained in curried.h), which can represent instances of the class "curry" that can take particular parameters, where void can be used to indicate a unit argument aka empty application. When all arguments are left out, it just represents any instance of "curry".

There is also a file uncurry.h, containing a function uncurry (returning the wrapped type), and the template type uncurried_t (to retrieve the wrapped type, aka the template argument of curry).

Finally, currying.h simply includes all other header files.

I made everything constexpr and qualifier sensetive but did not worry about noexcept.

There is, of course, no use of std::function or any form of allocation or overhead.

About

Simple and surprisingly powerful currying implemented concisely in c++20

Topics

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published