[Function] requiring an exact match in function signature?
data:image/s3,"s3://crabby-images/a3cae/a3cae14df8bc5e6a8b2aa907396120d185a05a6d" alt=""
Hello, I'm wondering if there is a way to force boost::function to only accept functions/functors with the exact same signature. I need this because there are cases in my program where passing a function with a different but convertible signature is almost certainly an error, and it would really be nice if such uses triggered a compiler error. For instance, boost::function will readily convert a mutator function with this signature: T mutate(const T&) to a function with this signature: void mutate(T&) even though the two functions clearly have different semantics. I know I could force the signatures to be the same by simply using a plain function pointer, but I want to give the user the flexibility to pass in function objects (functors) as well. Any suggestions? Cheers, Nate. _________________________________________________________________ Got a phone? Get Hotmail & Messenger for mobile! http://go.microsoft.com/?linkid=9724464
data:image/s3,"s3://crabby-images/3cdde/3cdde99a33dd10faf821fade4b762c93ab4a4310" alt=""
Nathan Ridge a écrit :
Hello,
I'm wondering if there is a way to force boost::function to only accept functions/functors with the exact same signature.
I need this because there are cases in my program where passing a function with a different but convertible signature is almost certainly an error, and it would really be nice if such uses triggered a compiler error.
For instance, boost::function will readily convert a mutator function with this signature:
boost::function doesn't really "convert" anything. The boost::function<sig> objects simply define an operator() with the `sig' signature, which in turns calls the wrapped function object by means of a virtual function call. So if your function is callable from a `sig' context, it works even though that's not its exact signature, because the arguments specified are convertible to the real arguments and the real return type is convertible to the specified return type. It doesn't really make sense to worry about the exact signature in my opinion; Anyway, if you want to test the signature of a function object, just take the address of &T::operator() and check it is what you want. Of course that won't work if it's a template or if it is overloaded.
data:image/s3,"s3://crabby-images/a3cae/a3cae14df8bc5e6a8b2aa907396120d185a05a6d" alt=""
Anyway, if you want to test the signature of a function object, just take the address of &T::operator() and check it is what you want. Of course that won't work if it's a template or if it is overloaded.
Could you be kind enough to give an example of how I can do such a check? Thanks, Nate. _________________________________________________________________ Hotmail & Messenger are available on your phone. Try now. http://go.microsoft.com/?linkid=9724461
data:image/s3,"s3://crabby-images/3cdde/3cdde99a33dd10faf821fade4b762c93ab4a4310" alt=""
Nathan Ridge a écrit :
Anyway, if you want to test the signature of a function object, just take the address of &T::operator() and check it is what you want. Of course that won't work if it's a template or if it is overloaded.
Could you be kind enough to give an example of how I can do such a check?
This works on GCC, haven't tested with other compilers.
#include
data:image/s3,"s3://crabby-images/a3cae/a3cae14df8bc5e6a8b2aa907396120d185a05a6d" alt=""
Anyway, if you want to test the signature of a function object, just take the address of &T::operator() and check it is what you want. Of course that won't work if it's a template or if it is overloaded.
Could you be kind enough to give an example of how I can do such a check?
This works on GCC, haven't tested with other compilers.
[snip]
Thank you so much for this example code.
Not only does this allow me to enforce an exact match in the
function signature, it also allows me to avoid using boost::function
altogether (the only reason I was using it was in an attempt to
enforce the signature), thus avoiding the associated performance
overhead.
I also learned a lot about templates and metaprogramming while trying
to understand how it works!
I modified the code slightly to allow for both functors and plain
functions. As this is my first attempt at nontrivial metaprogramming,
I would very much appreciate any comments on my modifications:
template
participants (2)
-
Mathias Gaunard
-
Nathan Ridge