Using fold expressions for common tasks

In functional programming, fold is a standard operator that encapsulates a simple pattern of recursion for processing lists1. In the following sections, I explore usage of fold operator for some common operations on template parameter pack. The template parameter pack can be thought of as a heterogeneous sequence of types.

count

template<typename... Args>
consteval auto count() {
    return sizeof...(Args);
}

Above we use sizeof... operator available since C++11, see sizeof…

count_if

template<template <typename T> typename Predicate, typename... Args>
consteval auto count_if() -> size_t {
    return (0 + ... + Predicate<Args>{}());
}

The Predicate passed above needs to be a template type that defines a compile time (consteval) function call operator() -> bool operator. Above functions folds over sum + binary operator. Example of such a Predicate:

template<typename T>
struct is_signed_type {
    consteval bool operator ()() const {
        return std::is_signed_v<T>;
    }
};

Example usage of count_if:

// evaluate to 2 at compile time
count_if<is_signed_type, int, unsigned int, double>(); 

all_of

template<template <typename T> typename Predicate, typename... Args>
consteval auto all_of() -> bool {
    return (Predicate<Args>{}() && ...);
}

Folds over the binary and (&&) operator.

any_of

template<template <typename T> typename Predicate, typename... Args>
consteval auto any_of() -> bool {
    return (Predicate<Args>{}() || ...);
}

Folds over binary or (||) operator.

none_of

template<template <typename T> typename Predicate, typename... Args>
consteval auto none_of() -> bool {
    return !any_of<Predicate, Args...>();
}

for_each

template<template <typename T> typename Func, typename... Args>
consteval void for_each() {
    (Func<Args>{}(), ...);
}

The Func passed above needs to be a template type that defines a compile time (consteval) function call operator() -> void operator. Here, we use comma operator , as the binary operator to fold over. Example of such a Func:

template<typename T>
struct assert_arithmetic {
    consteval void operator ()() const {
        static_assert(std::is_arithmetic_v<T>, "T must be arithmetic type");
    }
};

Traversing tuple types

When tuple is passed as input argument

template<template <typename T> typename Func, typename... Args>
consteval void for_each_tuple_type(std::tuple<Args...>) {
    (Func<Args>{}(), ...);
}

I apply fold over comma operator , to call Func for each type in the tuple. This function however requires passing the tuple as input argument even though the argument is unused. The argument only serves to deduce the types in the tuple.

When tuple is passed as template argument

template<template <typename T> typename Func, template Tuple>
consteval void for_each_tuple_type() {
    auto evaluator = []<typename Tuple, template <typename T> typename Func, size_t... I>(std::index_sequence<I...>) {
        (Func<std::tuple_element_t<I, Tuple>>{}(), ...);
    };
    evaluator.template operator() <Tuple, Func>
    (std::make_index_sequence<std::tuple_size_v<Tuple>>());
}

This function uses C++20 feature for template lambdas. The template lambda function here is useful to pass an index sequence to index into tuple to get the types. The fold operation remains same as before, i.e., fold over comma , operator.

Explore the example usage of all above helpers on compiler explorer.

References

  1. A tutorial on the universality and expressiveness of fold
  2. Not getting lost in translations - Daniela Engert - NDC TechTown 2024