I defined the output of literal_string as std:: string:
namespace parsesss {
namespace x3 = boost::spirit::x3;
auto const literal_string = x3::rule<class literal_string,std::string>{"literal_string"};
auto const literal_string_def = x3::lexeme['"' >> +(x3::char_ - '"') >> '"'];
BOOST_SPIRIT_DEFINE(literal_string);
}
Then I tested with the following code and the results also met expectations:
std::string test = "\"asdfjalsdjflajsdlfjalsdf\" \"xxxxxxxxxxasdfjalsdjflajsdlfjalsdf\"";
std::vector<std::string> out;
bool r = parse::x3::phrase_parse(test.begin(), test.end(), *(parsesss::literal_string), parse::x3::ascii::space, out);
if(r && (parse::g_iter == parse::g_end)){
std::cout << "parse success!" << std::endl;
}else{
std::cerr << "parse fail!" << std::endl;
}
But when I use variant data structures to store output, std:: string degenerates into char, why is this。
struct constant : x3::variant<long,char,double,float,int>{
using base_type::base_type;
using base_type::operator=;
};
struct last : x3::variant<std::string,char,constant>{
using base_type::base_type;
using base_type::operator=;
};
//
std::string test = "\"asdfjalsdjflajsdlfjalsdf\" \"xxxxxxxxxxasdfjalsdjflajsdlfjalsdf\"";
std::vector<last> out;
bool r = parse::x3::phrase_parse(test.begin(), test.end(), *(parsesss::literal_string), parse::x3::ascii::space, out);
if(r && (parse::g_iter == parse::g_end)){
std::cout << "parse success!" << std::endl;
}else{
std::cerr << "parse fail!" << std::endl;
}
Who can tell me why,
Or where can I find more detailed questions about this aspect on the official website。
The variant attribute can be constructed from a
char
, which is preferred.If you drop
char
from the variant element type list, it will work:Live On Coliru
Prints
Attempts
I tried hinting the attribute system with traits:
But, not unexpectedly, that didn't work.
Working Workarounds
What you can do instead is to break the compatibility of "char":
This already works (live). You can do shorter here: (live as well)
However, I noticed that these will just give you other trouble when you do want to parse the char literals...
Solution
So instead, I took the flight forwards and made some rules to match all the AST nodes.
As a rule of thumb, being explicit about attribute congruence helps. For example, the "naive" expression for
constant
will not work:For two reasons:
double_
will eagerly consume integral numbers. We avoid that by substitutingstrict_real_policies
x3::int_
is ambiguously compatible with eitherdouble
orintmax_t
. Now, there will be anx3
numeric parser likex3::long_long
that happens to matchintmax_t
, but it's better to be exactLive On Coliru
Printing