Следующая вспомогательная функция возвращает первое значение, если оно является rvalue, в противном случае второе значение (которое может быть rvalue, но может и не быть).
template <class T1, class T2>
typename std::enable_if<! std::is_reference<T1>::value, T1&&>::type
get_rvalue(T1&& t1, T2&& t2) { return std::forward<T1>(t1); }
template <class T1, class T2>
typename std::enable_if<std::is_reference<T1>::value, T2&&>::type
get_rvalue(T1&& t1, T2&& t2) { return std::forward<T2>(t2); }
Следующая вспомогательная функция возвращает другое значение, не возвращенное выше.
template <class T1, class T2>
typename std::enable_if<! std::is_reference<T1>::value, T1&&>::type
get_non_rvalue(T1&& t1, T2&& t2) { return std::forward<T2>(t2); }
template <class T1, class T2>
typename std::enable_if<std::is_reference<T1>::value, T2&&>::type
get_non_rvalue(T1&& t1, T2&& t2) { return std::forward<T1>(t1); }
Это просто сравнение, если два типа одинаковы, игнорируя ссылки и const.
template <class T1, class T2>
struct is_same_decay : public std::is_same<
typename std::decay<T1>::type,
typename std::decay<T2>::type
> {};
Тогда мы можем сделать только одну перегрузку для каждой функции (используя шаблоны), как показано ниже:
// 2ary ops
template <class M1, class M2>
friend typename std::enable_if<
is_same_decay<M1, Matrix>::value &&
is_same_decay<M2, Matrix>::value,
Matrix>::type
operator+(M1&& a, M2&& b)
{
Matrix x = get_rvalue(std::forward<M1>(a), std::forward<M2>(b));
x += get_non_rvalue(std::forward<M1>(a), std::forward<M2>(b));
return x;
}
template <class M1, class M2>
friend typename std::enable_if<
is_same_decay<M1, Matrix>::value &&
is_same_decay<M2, Matrix>::value,
Matrix>::type
operator*(M1&& a, M2&& b)
{
Matrix x = get_rvalue(std::forward<M1>(a), std::forward<M1>(b));
x *= get_non_rvalue(std::forward<M1>(a), std::forward<M1>(b));
return x;
}
Примечание выше, если M1
или M2
является r-значением, get_rvalue(a, b)
возвратит r-значение, следовательно, в этом случае Matrix x
будет заполнено движением, а не копией. Оптимизация именованного возвращаемого значения, вероятно, обеспечит отсутствие необходимости в копировании (или даже перемещении) возвращаемого значения, поскольку вместо возвращаемого значения будет создан x
.
Полный код здесь .