Reputation: 21520
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
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