Sugeriría usar sizeof(size_t)(o sizeof(ptrdiff_t)) que devuelve un tamaño "típico" relacionado con su máquina con la esperanza de que cualquier variable de este tamaño encaje en un registro. En ese caso, puede pasarlo de forma segura por valor. Además, como lo sugiere @ n314159 (ver comentarios al final de esta publicación), es útil asegurarse de que la variable también lo sea trivialy_copyable.
Aquí hay una demostración de C ++ 17:
#include <array>
#include <ccomplex>
#include <iostream>
#include <type_traits>
template <typename T>
struct maybe_ref
{
using type = std::conditional_t<sizeof(T) <= sizeof(size_t) and
std::is_trivially_copyable_v<T>, T, const T&>;
};
template <typename T>
using maybe_ref_t = typename maybe_ref<T>::type;
template <typename T>
class Foo
{
public:
Foo(maybe_ref_t<T> t) : _t(t)
{
std::cout << "is reference ? " << std::boolalpha
<< std::is_reference_v<decltype(t)> << std::endl;
}
private:
const T _t;
};
int main()
{
// with my machine
Foo<std::array<double, 1>> a{std::array<double, 1>{}}; // <- by value
Foo<std::array<double, 2>> b{std::array<double, 2>{}}; // <- by ref
Foo<double> c{double{}}; // <- by value
Foo<std::complex<double>> d{std::complex<double>{}}; // <- by ref
}