Luca Davanzo
Luca Davanzo

Reputation: 21520

C++: Call of overloaded ‘basic_string()’ is ambiguous

I have to post you my code because I can't get out on why I get this error!
I know I could have use templates..

class Mappable {
    typedef boost::variant<int, bool, unsigned short, float, char, timeval,
            double, std::string, size_t> MultiType;
private:
    class Handler {
        friend class Mappable;
    public:
        template<typename T>
        operator T&() {
            T t = 0;
            try {
                t = boost::get<T>(it);
            } catch (...) {
            }
            return t;
        }
        template<class T>
        Handler& operator=(const T& rhs) {
            it = rhs;
            return *this;
        }
    private:
        Handler(MultiType& it, const std::string& key);
        MultiType& it;
    };
public:
    Mappable(const std::string& tableName);
    virtual ~Mappable();
    Handler operator[](const std::string& key) {
        return Handler(map_[key], key);
    }
    std::string keyTypeToString(const std::string& key) {
        std::stringstream ss;
        ss << boost::get<T>(map_[key]);
        return ss.str();
    }
private:
    typedef std::map<std::string, MultiType> MultiTypeMap;
    std::string valueFromKey(const MultiTypeMap::iterator& it);
    template<class T> 
    MultiTypeMap map_;
};

/* Main.cpp */

int main() {
   Mappable m;
   m["x"] = 2;
   m["data"] = "my data string value"; /* Correctly works */
   cout << (int)(m["x"]); /* Correctly works */
   cout << (string)(m["data"]); /* Error */
   cout << m.keyTypeToString<string>("data"); /* Correctly works */
}

Error are:

../src/data/Mappable.h:98:41: error: call of overloaded ‘basic_string(Mappable::Handler)’ is ambiguous
../src/data/Mappable.h:98:41: note: candidates are:
/usr/include/c++/4.6/bits/basic_string.tcc:214:5: note: std::basic_string<_CharT, _Traits, _Alloc>::basic_string(const _CharT*, const _Alloc&) [with _CharT = char, _Traits = std::char_traits<char>, _Alloc = std::allocator<char>]
/usr/include/c++/4.6/bits/basic_string.tcc:171:5: note: std::basic_string<_CharT, _Traits, _Alloc>::basic_string(const std::basic_string<_CharT, _Traits, _Alloc>&) [with _CharT = char, _Traits = std::char_traits<char>, _Alloc = std::allocator<char>, std::basic_string<_CharT, _Traits, _Alloc> = std::basic_string<char>]
/usr/include/c++/4.6/bits/basic_string.tcc:179:5: note: std::basic_string<_CharT, _Traits, _Alloc>::basic_string(const _Alloc&) [with _CharT = char, _Traits = std::char_traits<char>, _Alloc = std::allocator<char>]

I can't understand where is ambiguity!

Upvotes: 1

Views: 2785

Answers (1)

sehe
sehe

Reputation: 393134

You have implicit conversions to any type.

template<typename T> operator T&();

This means, any single-param implicit constructor for basic_string is viable (char const*, std::string etc.)

The mantra is: implicit conversions are evil.

In C++11 you can add the explicit keyword to the conversion. This implies that you'll be invoking a static_cast<> explicitly

    template<typename T>
    explicit operator T&() { // throws on type mismatch
        return boost::get<T>(it);
    }

Upvotes: 4

Related Questions