Я предлагаю вам решение C ++ 14, которое должно быть адаптировано для C ++ 11 с заменами std::index_sequence
и std::make_index_sequence
.
Я предлагаю, для applyAsDoubled()
, просто вызвать вспомогательную функцию, передавая также std::index_sequence
над количеством массивов
template <typename F, std::size_t N, typename ... As>
void applyAsDoubled (F f, As(&...as)[N])
{ applyAsDoubledH(f, std::make_index_sequence<sizeof...(As)>{}, as...); }
Вспомогательная функция в значительной степени основана на std::tuple
, чтобы упаковать копии массивов, и новые std::array
, для копий массивов в стиле C
void applyAsDoubledH (F f, std::index_sequence<Is...> const & is,
Args(&...args)[N])
{
auto isn { std::make_index_sequence<N>{} };
std::tuple<std::array<Args, N>...> tpl { getStdArray(args, isn)... };
doublerMutation(tpl, is);
for (auto ui { 0u } ; ui < N ; ++ui )
f(std::get<Is>(tpl)[ui]...);
}
Наблюдайте за звонками на getStdArray()
template <typename T, std::size_t N, std::size_t ... Is>
std::array<T, N> getStdArray (T(&a)[N], std::index_sequence<Is...> const &)
{ return { { a[Is]... } }; }
для получения синглов std::array
форма одиночного массива в стиле C.
doublerMutation()
также использует вспомогательную функцию
template <std::size_t I, std::size_t N, typename ... Args>
void doublerMutationH (std::tuple<std::array<Args, N>...> & tpl)
{
for ( auto ui { 0u } ; ui < N ; ++ui )
std::get<I>(tpl)[ui] *= 2;
}
template <std::size_t N, typename ... Args, std::size_t ... Is>
void doublerMutation (std::tuple<std::array<Args, N>...> & tpl,
std::index_sequence<Is...> const &)
{
using unused = int[];
(void) unused { 0, (doublerMutationH<Is>(tpl), 0)... };
}
Ниже приведен полный рабочий пример
#include <tuple>
#include <array>
#include <iostream>
#include <type_traits>
template <std::size_t I, std::size_t N, typename ... Args>
void doublerMutationH (std::tuple<std::array<Args, N>...> & tpl)
{
for ( auto ui { 0u } ; ui < N ; ++ui )
std::get<I>(tpl)[ui] *= 2;
}
template <std::size_t N, typename ... Args, std::size_t ... Is>
void doublerMutation (std::tuple<std::array<Args, N>...> & tpl,
std::index_sequence<Is...> const &)
{
using unused = int[];
(void) unused { 0, (doublerMutationH<Is>(tpl), 0)... };
}
template <typename T, std::size_t N, std::size_t ... Is>
std::array<T, N> getStdArray (T(&a)[N], std::index_sequence<Is...> const &)
{ return { { a[Is]... } }; }
template <typename F, std::size_t ... Is, std::size_t N, typename ... Args>
void applyAsDoubledH (F f, std::index_sequence<Is...> const & is,
Args(&...args)[N])
{
auto isn { std::make_index_sequence<N>{} };
std::tuple<std::array<Args, N>...> tpl { getStdArray(args, isn)... };
doublerMutation(tpl, is);
for (auto ui { 0u } ; ui < N ; ++ui )
f(std::get<Is>(tpl)[ui]...);
}
template <typename F, std::size_t N, typename ... As>
void applyAsDoubled (F f, As(&...as)[N])
{ applyAsDoubledH(f, std::make_index_sequence<sizeof...(As)>{}, as...); }
int main ()
{
int A[] = { 1, 2, 3, 4, 5 };
long B[] = { 2, 2, 2, 2, 2 };
auto printSum = [](auto const & ... as)
{
using unused = int[];
typename std::common_type<decltype(as)...>::type sum {};
(void)unused { 0, (sum += as, 0)... };
std::cout << "the sum is " << sum << std::endl;
};
applyAsDoubled(printSum, A, B);
}
Если вы также можете использовать C ++ 17, используя свертывание шаблонов и оператор запятой, вы можете избежать использования массива unused
s, а doublerMutation()
можно упростить следующим образом:
template <std::size_t N, typename ... Args, std::size_t ... Is>
void doublerMutation (std::tuple<std::array<Args, N>...> & tpl,
std::index_sequence<Is...> const &)
{ ( doublerMutationH<Is>(tpl), ... ); }
и функция printSum()
лямбда-теста следующим образом
auto printSum = [](auto const & ... as)
{ std::cout << "the sum is " << (as + ...) << std::endl; };