具有两个参数的向量构造函数被解析为函数声明

时间:2012-01-19 14:41:03

标签: c++ parsing stdvector most-vexing-parse

考虑这个例子:

#include <iostream>
#include <string>
#include <vector>
#include <iterator>

int main()
{
    std::string sen = "abc def ghi jkl";
    std::istringstream iss(sen);

    std::vector<std::string>    // declaration in question
    vec(std::istream_iterator<std::string>(iss),
        std::istream_iterator<std::string>());

    std::copy(vec.begin(), vec.end(),
              std::ostream_iterator<std::string>(std::cout, "\n"));
}

编译器在调用std::copy

时抛出错误

request for member 'begin' in 'vec', which is of non-class type...

我可以解决这个错误:

std::istream_iterator<std::string> it_begin(iss);
std::istream_iterator<std::string> it_end;
std::vector<std::string> vec(it_begin, it_end);

或在每个参数周围加上括号,如下所示:

std::vector<std::string>
vec((std::istream_iterator<std::string>(iss)),
    (std::istream_iterator<std::string>()));

或者甚至在C ++ 11中使用新的统一初始化:

std::vector<std::string> vec { /*begin*/, /*end*/ };

为什么编译器将示例中的声明解析为函数声明?我知道最烦恼的解析,但我认为只有空参数列表才会发生。 我也想知道为什么第二种解决方法有效。

1 个答案:

答案 0 :(得分:9)

这仍然是最令人烦恼的解析。

std::vector<std::string>                     // return type
vec(                                         // function name
    std::istream_iterator<std::string>(iss), // param 1: an iterator called (iss), or just iss
    std::istream_iterator<std::string>()     // param 2: unnamed function 
);                                           //          returning iterator

geordi说:

<tomalak> << ETYPE_DESC(vec); std::vector<std::string> vec(std::istream_iterator<std::string>(iss), std::istream_iterator<std::string>());
<geordi> lvalue function taking a istream_iterator<string, char, char_traits<char>, long> , a pointer to a nullary function returning a istream_iterator<string, char, char_traits<char>, long> , and returning a vector of strings

它的关键在于,您的参数名称可以在它们周围加上括号(即iss(iss)),而不会改变声明的语义。有时。

使用另一组括号,如图所示,强制将第一个参数(以及第二个参数)解析为表达式而不是声明。< / p>


如果有帮助,也请考虑:

void foo(int (x)) {
   cout << x;
}

int main() {
   foo(42);
}

Output is 42