problem:I need to prevent given operator from being generated
if underlying operation can't be compiled. Basically, this operator==:
data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
template<class T> class accessor
{
typedef accessor<T> Self;
public:
template<class C> friend inline bool operator==(Self l, C& r)
data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{ return l.m_ref == r; }
}
data:image/s3,"s3://crabby-images/54783/547830fede928f19a3ce63b212a632c66666c748" alt=""
should not be instantiated if 'l.m_ref == r' fails to compile due to, for example, absence of related operator==.
solution:
template <bool Enable, typename T = void>
data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
struct enable_if
{
typedef T type;
};
template <typename T>
data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
struct enable_if<false, T>
{
};
data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
namespace Details
{
struct Shim
data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
data:image/s3,"s3://crabby-images/61f13/61f1322e8cdce7379843ad2586ac540cf468c453" alt=""
{
Shim(
); // Can be implicitely constructed by
// everything (Not defined).
};
struct NoMatchType
data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
data:image/s3,"s3://crabby-images/61f13/61f1322e8cdce7379843ad2586ac540cf468c453" alt=""
{
};
// Only declared, never defined:
NoMatchType operator==(const Shim&, const Shim&);
template <typename T, typename U>
struct IsEqualComparableImpl
data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
data:image/s3,"s3://crabby-images/61f13/61f1322e8cdce7379843ad2586ac540cf468c453" alt=""
{
private:
static const T& t();
static const U& u();
static char Check(bool); // Expected result
static char (&Check(const NoMatchType&))[2]; // Fallback
public:
static const bool value = sizeof(Check(t() == u())) == 1;
};
}
data:image/s3,"s3://crabby-images/54783/547830fede928f19a3ce63b212a632c66666c748" alt=""
template <typename T, typename U>
data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
struct IsEqualComparable
{
static const bool value =
Details::IsEqualComparableImpl<T, U>::value;
};
//test example
data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
template<class T> class accessor
{
typedef accessor<T> Self;
public:
data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
accessor() : m_ref()
{}
template<class C> friend inline typename enable_if<IsEqualComparable<T, C>::value, bool>::type
data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
operator==(Self l, C& r)
{
return l.m_ref == r;
}
private:
T m_ref;
};
data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
class C
{};
data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
class D
{};
data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
bool operator==(D, D)
{ return true; }
data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
bool test()
{
accessor<D> a1;
D d;
bool b = a1 == d; // OK
accessor<int> a;
C c;
return a == c; // Error
}
data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
int main()
{
return test();
}
data:image/s3,"s3://crabby-images/54783/547830fede928f19a3ce63b212a632c66666c748" alt=""
http://groups.google.com/group/comp.lang.c++.moderated/browse_thread/thread/3aa0bd9ea133b967/9ca97b1308becafe#9ca97b1308becafe