std::function as template parameter

前端 未结 2 1665
后悔当初
后悔当初 2021-02-07 12:35

I currently have a map, but for flexibility, I want to be able to assign a lambda expression, returning std::wstring as value

相关标签:
2条回答
  • 2021-02-07 12:42

    The basic problem is that std::function has a greedy implicit constructor that will attempt to convert anything, and only fail to compile in the body. So if you want to overload with it, either no conversion to the alternative can be allowed, of you need to disable stuff that can convert to the alternative from calling the std::function overload.

    The easiest technique would be tag dispatching. Make an operator= that is greedy and set up for perfect forwarding, then manually dispatch to an assign method with a tag:

     template<typename U>
     void operator=(U&&u){
       assign(std::forward<U>(u), std::is_convertible<U, std::wstring>());
     }
     void assign(std::wstring, std::true_type /*assign_to_string*/);
     void assign(std::function<blah>, std::false_type /*assign_to_non_string*/);
    

    basically we are doing manual overload resolution.

    More advanced techniques: (probably not needed)

    Another approach would be to limit the std::function = with SFINAE on the argument being invoked is valid, but that is messier.

    If you have multiple different types competing with your std::function you have to sadly manually dispatch all of them. The way to fix that is to test if your type U is callable with nothing and the result convertible to T, then tag dispatch on that. Stick the non-std::function overloads in the alternative branch, and let usual more traditional overloading to occur for everything else.

    There is a subtle difference in that a type convertible to both std::wstring and callable returning something convertible to T ends up being dispatched to different overloads than the original simple solution above, because the tests used are not actually mutually exclusive. For full manual emulation of C++ overloading (corrected for std::functions stupidity) you need to make that case ambiguous!

    The last advanced thing to do would be to use auto and trailing return types to improve the ability of other code to detect if your = is valid. Personally, I would not do this before C++14 except under duress, unless I was writing some serious library code.

    0 讨论(0)
  • 2021-02-07 13:00

    Both std::function and std::wstring have conversion operators that could take the literal wide string you are passing. In both cases the conversions are user defined and thus the conversion sequence takes the same precedence, causing the ambiguity. This is the root cause of the error.

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