What is the best way to convert a std::string to bool? I am calling a function that returns either "0" or "1", and I need a clean solution for turning this into a boolean value.
13 Answers
I am surprised that no one mentioned this one:
bool b; istringstream("1") >> b; or
bool b; istringstream("true") >> std::boolalpha >> b; 3 Comments
It'll probably be overkill for you, but I'd use boost::lexical_cast
boost::lexical_cast<bool>("1") // returns true boost::lexical_cast<bool>("0") // returns false 8 Comments
boost::bad_lexical_cast.Either you care about the possibility of an invalid return value or you don't. Most answers so far are in the middle ground, catching some strings besides "0" and "1", perhaps rationalizing about how they should be converted, perhaps throwing an exception. Invalid input cannot produce valid output, and you shouldn't try to accept it.
If you don't care about invalid returns, use s[0] == '1'. It's super simple and obvious. If you must justify its tolerance to someone, say it converts invalid input to false, and the empty string is likely to be a single \0 in your STL implementation so it's reasonably stable. s == "1" is also good, but s != "0" seems obtuse to me and makes invalid => true.
If you do care about errors (and likely should), use
if ( s.size() != 1 || s[0] < '0' || s[0] > '1' ) throw input_exception(); b = ( s[0] == '1' ); This catches ALL errors, it's also bluntly obvious and simple to anyone who knows a smidgen of C, and nothing will perform any faster.
Comments
DavidL's answer is the best, but I find myself wanting to support both forms of boolean input at the same time. So a minor variation on the theme (named after std::stoi):
bool stob(std::string s, bool throw_on_error = true) { auto result = false; // failure to assert is false std::istringstream is(s); // first try simple integer conversion is >> result; if (is.fail()) { // simple integer failed; try boolean is.clear(); is >> std::boolalpha >> result; } if (is.fail() && throw_on_error) { throw std::invalid_argument(s.append(" is not convertable to bool")); } return result; } This supports "0", "1", "true", and "false" as valid inputs. Unfortunately, I can't figure out a portable way to also support "TRUE" and "FALSE"
6 Comments
std::boolalpha stream extractor (which in turn relies on your locale). If lower-casing the string works for you, great.std::string lower = s; std::transform( lower.begin(), lower.end(), lower.begin(), ::tolower ); std::istringstream is( lower ); ... <continue on with your previous logic>Write a free function:
bool ToBool( const std::string & s ) { return s.at(0) == '1'; } This is about the simplest thing that might work, but you need to ask yourself:
- what should an empty string return? the version above throws an exception
- what should a character other than '1' or '0' convert to?
- is a string of more than one character a valid input for the function?
I'm sure there are others - this is the joy of API design!
7 Comments
at as the check is very clean.at().bool to_bool(std::string const &string) { return string[0] == '1'; } 1 Comment
I'd change the ugly function that returns this string in the first place. That's what bool is for.
3 Comments
std::string of all things for a boolean return?Here's a way similar to Kyle's except it handles the leading zeroes and stuff:
bool to_bool(std::string const& s) { return atoi(s.c_str()); } 2 Comments
You could always wrap the returned string in a class that handles the concept of boolean strings:
class BoolString : public string { public: BoolString(string const &s) : string(s) { if (s != "0" && s != "1") { throw invalid_argument(s); } } operator bool() { return *this == "1"; } } Call something like this:
BoolString bs(func_that_returns_string()); if (bs) ...; else ...; Which will throw invalid_argument if the rule about "0" and "1" is violated.
2 Comments
Try this:
bool value; if(string == "1") value = true; else if(string == "0") value = false;