0

我在 C++ 中尝试 noskipws 操纵器,并编写了以下代码。

#include <iostream>
#include <sstream>
#include <string>
using namespace std;

int main()
{
    string first, middle, last;

    istringstream("G B Shaw") >> first >> middle >> last;
    cout << "Default behavior: First Name = " << first << ", Middle Name = " << middle << ", Last Name = " << last << '\n';
    istringstream("G B Shaw") >> noskipws >> first >> middle >> last;
    cout << "noskipws behavior: First Name = " << first << ", Middle Name = " << middle << ", Last Name = " << last << '\n';
}

我期望以下输出:

预期产出

Default behavior: First Name = G, Middle Name = B, Last Name = Shaw
noskipws behavior: First Name = G, Middle Name = , Last Name = B

输出

Default behavior: First Name = G, Middle Name = B, Last Name = Shaw
noskipws behavior: First Name = G, Middle Name = , Last Name = Shaw

我修改了这段代码以使其适用于这样的字符,并且效果很好。

#include <iostream>
#include <sstream>
#include <string>
using namespace std;

int main()
{
    char first, middle, last;

    istringstream("G B S") >> first >> middle >> last;
    cout << "Default behavior: First Name = " << first << ", Middle Name = " << middle << ", Last Name = " << last << '\n';
    istringstream("G B S") >> noskipws >> first >> middle >> last;
    cout << "noskipws behavior: First Name = " << first << ", Middle Name = " << middle << ", Last Name = " << last << '\n';
}

我知道 cin 是如何工作的,但我无法弄清楚为什么在string.

4

3 回答 3

5
std::istringstream("G B S") >> std::noskipws >> first >> middle >> last;

对字符串执行提取时,首先清除字符串并将字符插入其缓冲区。

21.4.8.9 插入器和提取器

 template<class charT, class traits, class Allocator>
 basic_istream<charT, traits>&
     operator>>(basic_istream<charT, traits>& is,
                basic_string<charT, traits, Allocator>& str);

效果:表现为格式化输入函数 (27.7.2.2.1)。构造哨兵对象后,如果哨兵转换为true,则调用str.erase()然后从中提取字符is并将它们附加到,str就好像通过调用str.append(1, c).[...]

第一次读取会将字符串提取"G"first. 对于第二次提取,不会提取任何内容,因为std::noskipws设置了格式标志,禁用了前导空格的清除。因此,字符串被清除,然后提取失败,因为没有输入任何字符。这是上述子句的延续:

21.4.8.9 插入器和提取器(续)

[...] 字符被提取和附加,直到发生以下任何情况:

  • n存储字符;

  • 文件结束出现在输入序列上;

  • isspace(c, is.getloc())对于下一个可用的输入字符为真c

当流确定提取失败时,std::ios_base::failbit将设置为流状态,指示错误。

从此时起,除非流状态被清除,否则任何和所有 I/O 尝试都将失败。提取器无法运行,并且在未清除所有错误的流状态下将无法运行。这意味着提取到last不做任何事情,它保留了它在前一次提取时的值(没有 的那个std::noskipws),因为流没有清除字符串。


至于使用的原因char:字符在C或C++中没有格式要求。任何和所有字符都可以提取到 type 的对象中,这就是尽管设置char了但您仍然看到正确输出的原因:std::noskipws

27.7.2.2.3/1 [istream::extractors]

template<class charT, class traits>
basic_istream<charT, traits>& operator>>(basic_istream<charT, traits>& in,
                                         charT& c);

效果:行为类似于 in 的格式化输入成员(如 27.7.2.2.1 中所述)。在构建哨兵对象后in,如果有可用的字符,则从 中提取字符,并将其存储在c. 否则,函数调用in.setstate(failbit).

如果一个字符可用,提取器的语义会将一个字符存储到它的操作数中。它不界定空白(甚至是 EOF 字符!)。它将像普通字符一样提取它。

于 2014-02-16T23:54:49.280 回答
2

字符串的基本算法>>是:

1) 跳过空格
2) 读取并提取直到下一个空格

如果你使用noskipws,那么first step is skipped.

第一次读取后,您将定位在空白处,因此下一次(以及所有后续)读取将立即停止,不提取任何内容。
有关更多信息,您可以查看

表格cplusplus.com ,

许多提取操作将空格本身视为终止字符,因此,在禁用 skipws 标志的情况下,某些提取操作可​​能根本不从流中提取任何字符。
因此,在与字符串一起使用时,请删除 noskipws。

于 2014-02-16T10:23:34.490 回答
1

原因是在第二个示例中,您根本没有读取最后一个变量,而是打印它的旧值。

std::string first, middle, last;
std::istringstream iss("G B S");
                           ^^^
iss >> first >> middle >> last;
std::cout << "Default behavior: First Name = " << first 
            << ", Middle Name = " << middle << ", Last Name = " << last << '\n';
std::istringstream iss2("G B T");
                            ^^^
iss2 >> std::noskipws >> first >> middle >> last;
std::cout << "noskipws behavior: First Name = " << first 
            << ", Middle Name = " << middle << ", Last Name = " << last << '\n';

默认行为:名字 = G,中间名 = B,姓氏 = S

noskipws 行为:名字 = G,中间名 = ,姓氏 = S

发生这种情况是因为在第二次读取变量后,最后一个流位于空白处。

于 2014-02-16T10:43:58.857 回答