Why doesn't C++ have a pointer to member function type?

后端 未结 9 1798
一生所求
一生所求 2020-12-30 07:24

I could be totally wrong here, but as I understand it, C++ doesn\'t really have a native \"pointer to member function\" type. I know you can do tricks with Boost and mem_fu

相关标签:
9条回答
  • 2020-12-30 08:04

    I think what you are looking for might be in these librairies...

    Fast Delegates http://www.codeproject.com/KB/cpp/FastDelegate.aspx

    Boost.Function http://www.boost.org/doc/libs/1_37_0/doc/html/function.html

    And here is a very complete explanation of function pointer related questions http://www.parashift.com/c++-faq-lite/pointers-to-members.html

    0 讨论(0)
  • 2020-12-30 08:05

    As others pointed out, C++ does have a member function pointer type.

    The term you were looking for is "bound function". The reason C++ doesn't provide syntax sugar for function binding is because of its philosophy of only providing the most basic tools, with which you can then build all you want. This helps keep the language "small" (or at least, less mind bogglingly huge).

    Similarly, C++ doesn't have a lock{} primitive like C#'s but it has RAII which is used by boost's scoped_lock.

    There is of course the school of thought that says you should add syntax sugar for everything that might be of use. For better or worse, C++ does not belong to that school.

    0 讨论(0)
  • 2020-12-30 08:07

    The issue is surely not the basics of having an object pointer and a function pointer in one easy-to-use package, because you could do this using a pointer and a thunk. (Such thunks are already used by VC++ on x86 to support pointers to virtual member functions, so that these pointers only take up 4 bytes.) You might end up with a lot of thunks, it's true, but people already rely on the linker to eliminate duplicate template instantiations -- I do, anyway -- and there's only so many vtable and this offsets you'll end up with in practice. The overhead would probably not be significant for any reasonably-sized program, and if you don't use this stuff then it won't cost you anything.

    (Architectures that traditionally use a TOC would store the TOC pointer in the function pointer part, rather than in the thunk, just as they would have to do already.)

    (This new type of object would not be exactly substitutable for a normal pointer to function, of course, because the size would be different. They would, however, be written the same at the call point.)

    The problem that I see is that of the calling convention: supporting pointers to functions this way might be tricky in the general case, because the generated code would have to prepare the arguments (this included) the same way without regard to the actual type of thing, function or member function, to which the pointer points.

    This is probably not a big deal on x86, at least not with thiscall, because you could just load ECX regardless and accept that if the calling function doesn't need it then it will be bogus. (And I think VC++ assumes ECX is bogus in this case anyway.) But on architectures that pass arguments for named parameters in registers to functions, you may end up with a fair amount of shuffling in the thunk, and if stack arguments are pushed left-to-right then you're basically stuffed. And this can't be fixed up statically, because in the limit there's no cross-translation-unit information.

    [Edit: MSalters, in a comment to rocketmagnet's post above, points out that if both object AND function are known, then the this offset and so on can be determined immediately. This totally didn't occur to me! But, with this in mind, I suppose there need only be stored the exact object pointer, maybe offset, and the exact function pointer. This makes thunks totally unnecessary -- I think -- but I'm pretty sure that the issues of pointing to member functions and non-member functions alike would remain.]

    0 讨论(0)
  • 2020-12-30 08:15

    C++ is already a big language, and adding this would have made it bigger. What you really want is even worse than just a bound member function, it's something closer to boost::function. You want to store both a void(*)() and a pair for callbacks. After all, the reason is that you want the give the caller a complete callback, and the callee should not care about the exact details.

    The size would likely be sizeof(void*)+sizeof(void(*)()). Pointers to member functions can be bigger, but that is because they are unbound. They need to deal with the possibility that youre' taking the address of a virtual function, for instance. However, a built-in bound-pointer-to-member-function type would not suffer from this overhead. It can resolve the exact function to be called at the moment of binding.

    This is not possible with a UDT. boost::function cannot discard the overhead of a PTMF when it binds the object pointer. You need to know understand structure of a PTMF, vtable, etcetera - all non-standard stuff. However, we might get there now with C++1x. Once it's in std::, it's fair game for compiler vendors. The standard library implementation itself is not portable (see e.g. type_info).

    You'd still want to have some nice syntax, I guess, even if a compiler vendor implements this in the library. I'd like std::function<void(*)()> foo = &myX && X::bar. (It doesn't clash with existing syntax, as X::bar is not an expression - only &X::bar is)

    0 讨论(0)
  • 2020-12-30 08:22

    It does.

    For example,

    int (Fred::*)(char,float)
    

    is a pointer to a member function of a class Fred that returns an int and takes a char and a float.

    0 讨论(0)
  • 2020-12-30 08:23

    @RocketMagnet - This is in response to your other question, the one which was labeled a duplicate. I'm answering that question, not this one.

    In general, C++ pointer to member functions can't portably be cast across the class hierarchy. That said you can often get away with it. For instance:

    #include <iostream>
    using std::cout;
    class A { public: int x; };
    class B { public: int y; };
    class C : public B, public A { public: void foo(){ cout << "a.x == " << x << "\n";}};
    
    int main() {
        typedef void (A::*pmf_t)();
        C c; c.x = 42; c.y = -1;
    
        pmf_t mf = static_cast<pmf_t>(&C::foo);
        (c.*mf)();
    }
    

    Compile this code, and the compiler rightly complains:

    $ cl /EHsc /Zi /nologo pmf.cpp
    pmf.cpp
    pmf.cpp(15) : warning C4407: cast between different pointer to member representations, compiler may generate incorrect code
    
    $
    

    So to answer "why doesn't C++ have a pointer-to-member-function-on-void-class?" is that this imaginary base-class-of-everything has no members, so there's no value you could safely assign to it! "void (C::)()" and "void (void::)()" are mutually incompatible types.

    Now, I bet you're thinking "wait, i've cast member-function-pointers just fine before!" Yes, you may have, using reinterpret_cast and single inheritance. This is in the same category of other reinterpret casts:

    #include <iostream>
    using std::cout;
    class A { public: int x; };
    class B { public: int y; };
    class C : public B, public A { public: void foo(){ cout << "a.x == " << x << "\n";}};
    class D { public: int z; };
    
    int main() {
        C c; c.x = 42; c.y = -1;
    
        // this will print -1
        D& d = reinterpret_cast<D&>(c);
        cout << "d.z == " << d.z << "\n";
    }
    

    So if void (void::*)() did exist, but there is nothing you could safely/portably assign to it.

    Traditionally, you use functions of signature void (*)(void*) anywhere you'd thing of using void (void::*)(), because while member-function-pointers don't cast well up and down the inheritance heirarchy, void pointers do cast well. Instead:

    #include <iostream>
    using std::cout;
    class A { public: int x; };
    class B { public: int y; };
    class C : public B, public A { public: void foo(){ cout << "a.x == " << x << "\n";}};
    
    void do_foo(void* ptrToC){
        C* c = static_cast<C*>(ptrToC);
        c->foo();
    }
    
    int main() {
        typedef void (*pf_t)(void*);
        C c; c.x = 42; c.y = -1;
    
        pf_t f = do_foo;
        f(&c);
    }
    

    So to your question. Why doesn't C++ support this sort of casting. Pointer-to-member-function types already have to deal with virtual vs non-virtual base classes, and virtual vs non-virtual member functions, all in the same type, inflating them to 4*sizeof(void*) on some platforms. I think because it would further complicate the implementation of pointer-to-member-function, and raw function pointers already solve this problem so well.

    Like others have commented, C++ gives library writers enough tools to get this done, and then 'normal' programmers like you and me should use those libraries instead of sweating these details.

    EDIT: marked community wiki. Please only edit to include relevant references to the C++ standard, and add in italic. (esp. add references to standard where my understanding was wrong! ^_^ )

    0 讨论(0)
提交回复
热议问题