template <class Rhs, class Ret=dont_care>
struct has_logical_not : public true_type-or-false_type
{};
Наследники:Если (i)<rhs
>типа<Rhs
>можно использовать в выражении<!rhs
>, и (ii)<Ret=dont_care
>или результат выражения<!rhs
>конвертируем в<Ret
>, то наследуется отtrue_type, в противном случае наследуется отfalse_type.
По умолчанию поведение<Ret=dont_care
>не проверяется на возвратное значение префикса<operator!
>. Если<Ret
>отличается от типа по умолчанию<dont_care
>, то значение возврата проверяется на конвертируемость в<Ret
>. Конвертируемое к<Ret
>означает, что возвращаемое значение оператора может быть использовано в качестве аргумента для функции, ожидающей<Ret
>:
void f(Ret);
Rhs rhs;
f(!rhs);
Если<Ret=void
>, то тип возврата проверяется точно<void
>.
Заголовок:<#include
<boost/type_traits/has_logical_not.hpp>
>или<#include<boost/type_traits/has_operator.hpp>
>или<#include<boost/type_traits.hpp>
>
Совместимость компилятора:Требуется работа SFINAE (т.е. BOOST_NO_SFINAE не установлен). Этого не поддерживает лишь меньшинство довольно старых компиляторов.
Примеры:
<has_logical_not<Rhs,Ret>::value_type
>является типом<bool
>.
<has_logical_not<Rhs,Ret>::value
>— интегральное постоянное выражение<bool
>.
<has_logical_not<int>::value
>— интегральное постоянное выражение<bool
>, оценивающее до<true
>.
<has_logical_not<bool>
>наследуется от<true_type
>.
<has_logical_not<int,bool>
>наследуется от<true_type
>.
<has_logical_not<int,long>
>наследуется от<true_type
>.
<has_logical_not<double,double>
>наследуется от<true_type
>.
<has_logical_not<double,bool>
>наследуется от<true_type
>.
<has_logical_not<constbool>
>наследуется от<true_type
>.
<has_logical_not<int,std::string>
>наследуется от<false_type
>.
См. также:Типы операторов
Известные вопросы:
- Эта черта не может определить, является ли префикс<
operator!
>общедоступным или нет: если<operator!
>определен как частный член<Rhs
>, то инстанцирование<has_logical_not<Rhs>
>приведет к ошибке компилятора. По этой причине<has_logical_not
>не может использоваться для определения того, имеет ли тип публичный<operator!
>или нет.<structA{private:voidoperator!();};
boost::has_logical_not<A>::value;
>
- Существует проблема, если оператор существует только для типа<
A
>и<B
>является конвертируемым в<A
>. В этом случае компилятор сообщит о двусмысленной перегрузке.<structA{};
voidoperator!(constA&);
structB{operatorA();};
boost::has_logical_not<A>::value;
boost::has_logical_not<B>::value;
>
- Существует проблема при применении этой черты к классам шаблонов. Если<
operator!
>определено, но не связывается для данного типа шаблона, он все еще обнаруживается чертой, которая возвращается<true
>вместо<false
>. Пример:<#include<boost/type_traits/has_logical_not.hpp>
#include<iostream>
template<classT>
structcontains{Tdata;};
template<classT>
booloperator!(constcontains<T>&rhs){
returnf(rhs.data);
}
classbad{};
classgood{};
boolf(constgood&){}
intmain(){
std::cout<<std::boolalpha;
std::cout<<boost::has_logical_not<contains<good>>::value<<'\n';
contains<good>g;
!g;
std::cout<<boost::has_logical_not<contains<bad>>::value<<'\n';
contains<bad>b;
!b;
return0;
}
>
- <
volatile
>квалификатор не обрабатывается должным образом и приведет к неопределенному поведению