- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我正在编写程序的一部分,用于解析和验证程序控制台参数中的一些用户输入。为此,我选择使用 stringstream,但遇到了读取无符号类型的问题。
下一个模板用于从给定字符串中读取请求的类型:
#include <iostream>
#include <sstream>
#include <string>
using std::string;
using std::stringstream;
using std::cout;
using std::endl;
template<typename ValueType>
ValueType read_value(string s)
{
stringstream ss(s);
ValueType res;
ss >> res;
if (ss.fail() or not ss.eof())
throw string("Bad argument: ") + s;
return res;
}
// +template specializations for strings, etc.
int main(void)
{
cout << read_value<unsigned int>("-10") << endl;
}
如果类型是无符号的并且输入字符串包含负数,我希望看到异常抛出(由 ss.fail() = true
引起)。但是 stringstream 生成转换为无符号类型的值(书面示例中为 4294967286)。
如何修复此示例以实现所需的行为(最好不要回退到 c 函数)?我知道这可以通过简单的第一个符号检查来完成,但我可以放置前导空格。我可以编写自己的解析器,但不相信问题如此不可预测,标准库无法解决。
隐藏在无符号类型字符串流运算符深处的函数是 strtoull 和 strtoul。它们以描述的方式工作,但提到的功能是低级的。为什么 stringstream 不提供一些验证级别? (我只希望我是错的,它确实是错的,但需要一些 Action 才能实现)。
最佳答案
版本免责声明:C++03 的答案不同。以下内容涉及 C++11。
首先,让我们分析一下发生了什么。
ss >> res;
这称为 std::istream::operator>>(unsigned)
.在[istream.formatted.arithmetic]/1中,effects定义如下:
These extractors behave as formatted input functions (as described in 27.7.2.2.1). After a sentry object is constructed, the conversion occurs as if performed by the following code fragment:
typedef num_get< charT,istreambuf_iterator<charT,traits> > numget;
iostate err = iostate::goodbit;
use_facet< numget >(loc).get(*this, 0, *this, err, val);
setstate(err);In the above fragment,
loc
stands for the private member of thebasic_ios
class.
跟随格式化输入函数到[istream::sentry],sentry
的主要作用这里的目标是消耗前导空白字符。它还可以防止在出现错误时执行上面显示的代码(流处于失败/eof 状态)。
使用的语言环境是 "C"
语言环境。理由:
对于stringstream
通过 stringstream ss(s);
构建,该 iostream 的语言环境是构建时的当前全局语言环境(这在 [ios.base.locales]/4 的兔子洞深处得到保证)。由于 OP 程序中的全局区域设置未更改,因此 [locale.cons]/2 指定“经典”区域设置,即 "C"
语言环境。
use_facet< numget >(loc).get
使用成员函数 num_get<char>::get(iter_type in, iter_type end, ios_base&, ios_base::iostate& err, unsigned int& v) const;
在 [locale.num.get] 中指定(注意 unsigned int
,一切都还好)。字符串的详细信息 -> unsigned int
“C”语言环境的转换很长,在 [facet.num.get.virtuals] 中有描述。一些有趣的细节:
strtoull
被使用。ios_base::failbit
分配给 err
.具体来说:“要存储的数值可以是以下之一:[...] 最负的可表示值或无符号整数类型的零,如果该字段表示一个太大的负值而无法在 val 中表示。ios_base::failbit
分配给 err
。”strtoull
的定义需要到C99, 7.20.1.4 , 在第 5 段下:
If the subject sequence begins with a minus sign, the value resulting from the conversion is negated (in the return type).
第 8 段:
If the correct value is outside the range of representable values,
LONG_MIN
,LONG_MAX
,LLONG_MIN
,LLONG_MAX
,ULONG_MAX
, orULLONG_MAX
is returned (according to the return type and sign of the value, if any), and the value of the macroERANGE
is stored inerrno
对于strotoul
,负值是否被视为有效输入似乎在过去一直存在争议。 .无论如何,问题出在这个函数上。对 gcc 的快速检查表明它被认为是有效输入,因此您观察到的行为也是如此。
历史记录:C++03
使用 C++03 scanf
在num_get
里面转换。不幸的是,我不太确定(还)如何转换为 scanf
是指定的,在什么情况下会发生错误。
显式错误检查:
我们可以通过使用带符号的值进行转换和测试来手动插入该支票 <0
,或者我们寻找 -
字符(这不是一个好主意,因为可能存在本地化问题)。
关于c++ - stringstream 无符号输入验证,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18917167/
我重载了 operator", str); // googletest test //Second test ss.str("");ss.clear(); //Mandatory if 'ss' ne
我有这个代码,想知道是否有可能让 stringstream 将 uint8_t 视为数字而不是字符? uint8_t s; std::stringstream sstream( "255" ); ss
函数 fprintf 写入 FILE*。我有一个写入 stringstream 的 debugPrint 函数。我不想更改我的功能,因为它在很多地方都被使用,而且它会破坏代码。 如何将字符串流传递给
我知道 stringstream 可以用 stringstream::str() 更新,但是如果之后我在 stringstream 中输入其他内容,它没有按预期工作。以下片段演示了该现象: #incl
我希望 stringstream 有一个从 string&& 窃取其初始内容的构造函数。 STL中通常不存在这种跨物种的“move 构造函数”吗?如果没有,为什么不呢? 最佳答案 有历史,令人失望。但
我什么时候使用 std::istringstream、std::ostringstream 和 std::stringstream 以及为什么我不应该只使用std::stringstream 在每个场
考虑以下代码,它编译并运行: #include #include struct Foo {}; void operator>>(std::istream &, Foo) { } int main(
考虑: std::string s_a, s_b; std::stringstream ss_1, ss_2; // at this stage: // ss_1 and ss
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 这个问题似乎是题外话,因为它缺乏足够的信息来诊断问题。 更详细地描述您的问题或include a min
这个问题在这里已经有了答案: Why is "using namespace std;" considered bad practice? (42 个回答) 关闭 9 年前。 我是 C++ 的新手,
伙计们,实际上我想在这里做的是,当我输入 3,12,36 时,输出将是: 3 12 36 但是在这里我很难让它输出所有的答案。我一直在做的是,当您输入 3,12,36 时,它只会输出 3 12,如果您
我这里有这段代码 cout > s; stringstream is(s); is >> x >> y; cout > x cin >> y 关于C++ Strings
我遇到了以下代码的问题。 short int decrypted0[] = {0,2}; string message = "22"; string string0 =""; for(i=0;i
我正在尝试一个“stringstream”程序,它是这样的: #include #include using namespace std; int main() { int x; char ch;
我的印象是未格式化的 write()会将字节逐字复制到 ostream它被称为。另一位 SO 用户还告诉我“写入函数不会跳过任何内容”。如果您在使用 stringstream 时遇到问题,问一个问题。
这是一个用于解析文本文件的示例,问题是如果我在内部 while 循环之外声明“string parm”,那么一行中的最后一个字符串总是得到双重输出(这只发生在某些空格字符出现在“source.txt”
这个问题在这里已经有了答案: How to clear stringstream? [duplicate] (1 个回答) 关闭 9 年前。 例如我有这段代码: #include #include
#include #include #include #include using namespace std; stack aStack; s
我试图仅将 char 数组中的数字添加到 stringstream 对象中。代码是: char[50] buffer = ''; stringstream str; int page; str > p
为什么这个程序打印出空白? string str; stringstream ss(str); ss << "A string"; cout << str; // just blank
我是一名优秀的程序员,十分优秀!