C ++ 11: Liczba zmiennych parametrów funkcji szablonu?

87

Jak mogę uzyskać liczbę argumentów do zmiennej funkcji szablonu?

to znaczy:

template<typename... T>
void f(const T&... t)
{
    int n = number_of_args(t);

    ...
}

Jaki jest najlepszy sposób realizacji number_of_argspowyższego?

Andrew Tomazos
źródło
28
sizeof...(T).
R. Martinho Fernandes,
22
@ R.MartinhoFernandes formularz „Opublikuj odpowiedź” znajduje się kilka centymetrów dalej na dole strony. ;)
kay
@ kay-SEisevil Nie rozumiem, dlaczego twój komentarz ma mniej głosów poparcia niż jego.
Sapphire_Brick

Odpowiedzi:

104

Po prostu napisz to:

const std::size_t n = sizeof...(T); //you may use `constexpr` instead of `const`

Zauważ, że njest to wyrażenie stałe (znane w czasie kompilacji), co oznacza, że ​​możesz go użyć tam, gdzie potrzebne jest wyrażenie stałe, na przykład:

std::array<int,   n>  a; //array of  n elements
std::array<int, 2*n>  b; //array of (2*n) elements

auto middle = std::get<n/2>(tupleInstance);

Zauważ, że jeśli chcesz obliczyć zagregowany rozmiar spakowanych typów (w przeciwieństwie do liczby typów w paczce), musisz zrobić coś takiego:

template<std::size_t ...>
struct add_all : std::integral_constant< std::size_t,0 > {};

template<std::size_t X, std::size_t ... Xs>
struct add_all<X,Xs...> : 
  std::integral_constant< std::size_t, X + add_all<Xs...>::value > {};

następnie zrób to:

constexpr auto size = add_all< sizeof(T)... >::value;

W C ++ 17 (i nowszych) obliczanie sumy rozmiarów typów jest znacznie prostsze przy użyciu wyrażenia fold :

constexpr auto size = (sizeof(T) + ...);

Mam nadzieję, że to pomoże.

Nawaz
źródło
9
+1 Nauczyłem się dwóch rzeczy; sizeof...i constexpr. :)
Qix - MONICA ZOSTAŁA POMYŚLNA
1
Więc to sizeof...faktycznie zwraca liczbę argumentów, a nie łączny rozmiar wszystkich argumentów (jak sizeofw tablicy)?
panzi,
@panzi: Tak. sizeof ...(T)zwraca liczbę zapakowanych typów T. Jeśli chcesz obliczyć zagregowany rozmiar upakowanych typów, musisz zrobić coś takiego: ideone.com/udggBk Dodałem to również w mojej odpowiedzi.
Nawaz,
@panzi: obliczenia w mojej odpowiedzi są teraz nieco poprawione.
Nawaz,
2
W C ++ 17, aby obliczyć rozmiar poszczególnych typów argumentów, możemy teraz użyć wyrażeń fold return (0 + ... + sizeof(t));
P0W