C++中cin读取失败后原变量置为0的问题
今天某个同学提出了这个问题:使用cin将一个char传入到一个int中去,最终结果是多少?
#include<iostream>
int main(int argc, char* argv[])
{
int i;
std::cin >> i; //这里输入一个‘a’
std::cout << i;
return 0;
}
当即另有人回复曰:自然是转换为对应的ASCII码值。我也是这么猜的,结果很遗憾

随后又不信邪地把i初始化了一下试试
#include<iostream>
int main(int argc, char* argv[])
{
int i{ 5 };
std::cout << i << std::endl;
std::cin >> i; //这里输入一个‘a’
std::cout << i;
return 0;
}

结果还是0,甚至还把我之前初始化的值给盖掉了。
这时候我意识到这应该是某个我不清楚的机制了,在查阅C++11标准文案后找到了这么一段:
ISO/IEC 14882:2011 § 27.7.2.2.2 3
operator>>(int& val);
The conversion occurs as if performed by the following code fragment (using the same notation as for
the preceding code fragment):typedef num_get<charT,istreambuf_iterator<charT,traits> > numget; iostate err = ios_base::goodbit; long lval; use_facet<numget>(loc).get(*this, 0, *this, err, lval); if (lval < numeric_limits<int>::min()) { err |= ios_base::failbit; val = numeric_limits<int>::min(); } else if (numeric_limits<int>::max() < lval) { err |= ios_base::failbit; val = numeric_limits<int>::max(); } else val = static_cast<int>(lval); setstate(err);
在cppreference中也给出了阐述:
If extraction fails (e.g. if a letter was entered where a digit is expected), value is left unmodified and failbit is set. (until C++11)
If extraction fails, zero is written to value and failbit is set. If extraction results in the value too large or too small to fit in value, std::numeric_limits::max() or std::numeric_limits::min() is written and failbit flag is set. (since C++11)
———————————————————————————————————————————————
若释出失败(例如若在期待数位处遇到字母),则保留 value 不修改并设置 failbit 。(C++11 前)
若释出失败,则写入零到 value 并设置 failbit 。若释出结果对于 value 过大或过小,则写入 std::numeric_limits::max() 或 std::numeric_limits::min() 并设置 failbit 标志。(C++11 起)
参数
结合上述资料可以总结:cin在写入数据失败时,会将原变量置零,而且当溢出时,会写入std::numeric_limits::max() 或 std::numeric_limits::min(),此外当然还有我们所熟知的将failbit置位。而这是C++11增加的一个新特性,在以前的标准中是没有将变量置零那一步的。
搞定之后,甚觉自己C++基础之欠缺,编程之路,道阻且长!



674

被折叠的 条评论
为什么被折叠?



