has_member

C

Christof Warlich

Hi,

I just found the following code that shows how to determine whether a
class has a certain member:

#include <iostream>
typedef char (&no_tag)[1];
typedef char (&yes_tag)[2];

template <typename T, void (T::*)()> struct ptmf_helper {};
template<typename T> no_tag has_member_foo_helper(...);

template<typename T>
yes_tag has_member_foo_helper(ptmf_helper<T, &T::foo>* p);

template<typename T> struct has_member_foo {
static const bool value = sizeof(has_member_foo_helper<T>(0)) ==
sizeof(yes_tag);
};

struct HasNoFoo {};
struct HasFoo {void foo();};
struct DerivedFromHasFoo: HasFoo {};

int main() {
std::cout << has_member_foo<HasNoFoo>::value << std::endl;
std::cout << has_member_foo<HasFoo>::value << std::endl;
std::cout << has_member_foo<DerivedFromHasFoo>::value << std::endl;
}

But what I'd really need is some "provides_member" functionaliy, i.e.
something that would also evaluate to true for any derived class of
class HasFoo. Anyone having an idea how this could be accomplished?

Furthermore, is anyone aware whether something like has_member and
provides_member is available in Boost? I would really miss it as part of
Boost's type_traits, but couldn't find it.

Finally, can anyone recommend a good overview on the functionality
provided by Boost, most notably with respect to template metaprogramming?

Thanks,

Christof
 
C

courpron

Hi,

I just found the following code that shows how to determine whether a
class has a certain member:

#include <iostream>
typedef char (&no_tag)[1];
typedef char (&yes_tag)[2];

template <typename T, void (T::*)()> struct ptmf_helper {};
template<typename T> no_tag has_member_foo_helper(...);

template<typename T>
yes_tag has_member_foo_helper(ptmf_helper<T, &T::foo>* p);

template<typename T> struct has_member_foo {
     static const bool value = sizeof(has_member_foo_helper<T>(0)) ==
                               sizeof(yes_tag);

};

struct HasNoFoo {};
struct HasFoo {void foo();};
struct DerivedFromHasFoo: HasFoo {};

int main() {
     std::cout << has_member_foo<HasNoFoo>::value << std::endl;
     std::cout << has_member_foo<HasFoo>::value << std::endl;
     std::cout << has_member_foo<DerivedFromHasFoo>::value << std::endl;

}

But what I'd really need is some "provides_member" functionaliy, i.e.
something that would also evaluate to true for any derived class of
class HasFoo. Anyone having an idea how this could be accomplished?


Hi,

AFAIK, you can't in a truly generic way. Pointers to members as
template parameters don't accept any conversion. In some way or
another, you must provide the base class ( that actually declares the
member function ) as a template parameter to construct the right
function signature.

I explain everything related to member detection in this article :
http://www.codeproject.com/KB/architecture/Detector.aspx

Alexandre Courpron.
 
J

Jeff Schwab

Christof said:
Hi,

I just found the following code that shows how to determine whether a
class has a certain member:

#include <iostream>
typedef char (&no_tag)[1];
typedef char (&yes_tag)[2];

template <typename T, void (T::*)()> struct ptmf_helper {};
template<typename T> no_tag has_member_foo_helper(...);

template<typename T>
yes_tag has_member_foo_helper(ptmf_helper<T, &T::foo>* p);

template<typename T> struct has_member_foo {
static const bool value = sizeof(has_member_foo_helper<T>(0)) ==
sizeof(yes_tag);
};

struct HasNoFoo {};
struct HasFoo {void foo();};
struct DerivedFromHasFoo: HasFoo {};

int main() {
std::cout << has_member_foo<HasNoFoo>::value << std::endl;
std::cout << has_member_foo<HasFoo>::value << std::endl;
std::cout << has_member_foo<DerivedFromHasFoo>::value << std::endl;
}

But what I'd really need is some "provides_member" functionaliy, i.e.
something that would also evaluate to true for any derived class of
class HasFoo. Anyone having an idea how this could be accomplished?

Possibly, through some clever SFINAE, but nothing comes immediately to mind.
Furthermore, is anyone aware whether something like has_member and
provides_member is available in Boost? I would really miss it as part of
Boost's type_traits, but couldn't find it.

Finally, can anyone recommend a good overview on the functionality
provided by Boost, most notably with respect to template metaprogramming?

http://www.boost-consulting.com/tmpbook/
 
C

Christof Warlich

AFAIK, you can't in a truly generic way. Pointers to members as
template parameters don't accept any conversion. In some way or
another, you must provide the base class ( that actually declares the
member function ) as a template parameter to construct the right
function signature.
Ok, I understand that the prototypes of any base classes members have an
implicit this pointer of the type of the base, which makes its signature
different. Thanks for this insight. But this fact brings the idea to my
mind whether that could be worked around by always exchanging these this
pointers with some fixed pointer type and then comparing these
function prototypes? I'm just dreaming, as my limited skills in this
area currently do not allow for more.
I explain everything related to member detection in this article :
http://www.codeproject.com/KB/architecture/Detector.aspx
In any case, thanks a lot for this excellent article with its detailed
explanation and the generalization provided through the DETECTOR macros.
It's really a pity that it may not be possible to make it work for
members defined in parent classes, as it limits its use significantly.
 
C

courpron

(e-mail address removed) schrieb:> AFAIK, you can't in a truly generic way. Pointers to members as

Ok, I understand that the prototypes of any base classes members have an
implicit this pointer of the type of the base, which makes its signature
different. Thanks for this insight. But this fact brings the idea to my
mind whether that could be worked around by always exchanging these this
  pointers with some fixed pointer type and then comparing these
function prototypes? I'm just dreaming, as my limited skills in this
area currently do not allow for more.

You must use the exact signature, which includes the this pointer.
There is no escape (AFAIK).
In any case, thanks a lot for this excellent article with its detailed
explanation and the generalization provided through the DETECTOR macros.
It's really a pity that it may not be possible to make it work for
members defined in parent classes, as it limits its use significantly.

Thanks.
I agree this is an important limitation.

If you have Visual C++ (7.1 or higher version) and portability isn't
an issue, you can use the __if_exists keyword which allows it :

http://msdn2.microsoft.com/en-us/library/x7wy9xh3(VS.80).aspx

Alexandre Courpron.
 
C

Christof Warlich

Thanks.
I agree this is an important limitation.
Having slept a night over this, I came to the conclusion that it may not
be as usable as I initially thought: If it would be possible, what about
multiple, non-virtual inheritance? If the member being tested exists in
more than one base, then I need to know which one I may want to use,
which requires to know the type of that base.
If you have Visual C++ (7.1 or higher version) and portability isn't
an issue, you can use the __if_exists keyword which allows it :

http://msdn2.microsoft.com/en-us/library/x7wy9xh3(VS.80).aspx
Thanks for this link. But for the reasons pointed out above, I'm quite
comfortable now with your current solution. Again many many thanks for
your valuable contribution.
 
C

Christof Warlich

Jeff said:
Possibly, through some clever SFINAE, but nothing comes immediately to
mind.

In the mean time, I doubt that it may be worth the effort to further spend
time on this issue for the reasons pointed out in my previous answer to
Alexandre.

Thanks for this recommendation, it looks like the perfect match for my
needs as I just realize that I have taken my initial code example from a
post of one of the authors of the book, Aleksey Gurtovoy.
 
C

courpron

(e-mail address removed) schrieb:>> It's really a pity that it may not be possible to make it work for


Having slept a night over this, I came to the conclusion that it may not
be as usable as I initially thought: If it would be possible, what about
multiple, non-virtual inheritance? If the member being tested exists in
  more than one base, then I need to know which one I may want to use,
which requires to know the type of that base.



Thanks for this link. But for the reasons pointed out above, I'm quite
comfortable now with your current solution. Again many many thanks for
your valuable contribution.

You're welcome.

Alexandre Courpron.
 

Ask a Question

Want to reply to this thread or ask your own question?

You'll need to choose a username for the site, which only take a couple of moments. After that, you can post your question and our members will help you out.

Ask a Question

Members online

No members online now.

Forum statistics

Threads
474,176
Messages
2,570,947
Members
47,498
Latest member
log5Sshell/alfa5

Latest Threads

Top