- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
我在 Linux 上使用 GCC 4.9.0。这是我的测试程序:
#include <iostream>
#include <string>
using namespace std;
int main(int argc, char* argv[])
{
size_t pos = 42;
cout << "result: " << stoi(argv[1], &pos, atoi(argv[2])) << '\n';
cout << "consumed: " << pos << '\n';
}
这是一个预期的结果:
$ ./a.out 100 2
result: 4
consumed: 3
也就是说,它将以 2 为底的“100”解析为数字 4,并消耗了所有 3 个字符。
我们可以在 36 以内进行类似操作:
$ ./a.out 100 36
result: 1296
consumed: 3
但是更大的基地呢?
$ ./a.out 100 37
result: 0
consumed: 18446744073707449552
这是什么? pos
应该是它停止解析的索引。这里它接近 std::string::npos
但并不完全(相差几百万)。如果我在没有优化的情况下编译,那么 pos
是 18446744073703251929
,所以它看起来像未初始化的垃圾,尽管我确实初始化了它(到 42)。事实上,valgrind 提示道:
Conditional jump or move depends on uninitialised value(s)
at 0x400F11: int __gnu_cxx::__stoa<long, int, char, int>(...) (in a.out)
by 0x400EC7: std::stoi(std::string const&, unsigned long*, int) (in a.out)
所以这很有趣。此外,std::stoi
的文档说,如果无法执行转换,它会抛出 std::invalid_argument。显然这种情况下它没有进行任何转换,它在pos
中返回了垃圾,并且没有抛出异常。
如果 base
为 1 或负数,也会发生类似的坏事。
这是 GCC 实现中的错误、标准中的错误,还是我们必须学会忍受的东西?我认为 stoi()
与 atoi()
的目标之一是更好的错误检测,但似乎根本不检查 base
。
编辑:这是同一程序的 C 版本,它也打印 errno:
#include <stdio.h>
#include <stdlib.h>
int main(int argc, char* argv[])
{
char* pos = (char*)42;
printf("result: %ld\n", strtol(argv[1], &pos, atoi(argv[2])));
printf("consumed: %lu (%p)\n", pos - argv[1], pos);
perror("errno");
return 0;
}
当它工作时,它会做和以前一样的事情。当它失败时,它会更加清晰:
$ ./a.out 100 37
result: 0
consumed: 18446603340345143502 (0x2a)
errno: Invalid argument
现在我们明白了为什么 C++ 版本中的 pos
是一个“垃圾”值:这是因为 strtol()
没有改变 endptr
,并且 C++ 包装器错误地从中减去输入字符串的起始地址。
在 C 版本中,我们还看到 errno
设置为 EINVAL
以指示错误。我系统上的文档说当 base
无效时会发生这种情况,但也说它不是 C99 指定的。如果我们在 C++ 版本中打印 errno
我们也可以检测到这个错误(但它在 C99 中不是标准的,并且肯定不是 C++11 指定的)。
最佳答案
[C++11: 21.5/3]:
Throws: invalid_argument ifstrtol
,strtoul
,strtoll
, orstrtoull
reports that no conversion could be performed. [..]
[C99: 7.20.1.4/5]:
If the subject sequence has the expected form and the value ofbase
is zero, the sequence of characters starting with the first digit is interpreted as an integer constant according to the rules of 6.4.4.1. If the subject sequence has the expected form and the value ofbase
is between 2 and 36, it is used as the base for conversion, ascribing to each letter its value as given above. [..]
C99 中没有为 base
为 not 0 或介于 2 和 36 之间的情况指定语义,因此结果未定义。这不一定满足 [C++11: 21.5/3]
的摘录。
简而言之,这是 UB;只有当基数有效但输入值在该基数中不可转换时,您才会期望出现异常。 这既不是 GCC 也不是标准中的错误。
关于当 base 不在 [2,36] (GCC) 中时,C++11 std::stoi 静默失败,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24504269/
我需要将文本放在 中在一个 Div 中,在另一个 Div 中,在另一个 Div 中。所以这是它的样子: #document Change PIN
奇怪的事情发生了。 我有一个基本的 html 代码。 html,头部, body 。(因为我收到了一些反对票,这里是完整的代码) 这是我的CSS: html { backgroun
我正在尝试将 Assets 中的一组图像加载到 UICollectionview 中存在的 ImageView 中,但每当我运行应用程序时它都会显示错误。而且也没有显示图像。 我在ViewDidLoa
我需要根据带参数的 perl 脚本的输出更改一些环境变量。在 tcsh 中,我可以使用别名命令来评估 perl 脚本的输出。 tcsh: alias setsdk 'eval `/localhome/
我使用 Windows 身份验证创建了一个新的 Blazor(服务器端)应用程序,并使用 IIS Express 运行它。它将显示一条消息“Hello Domain\User!”来自右上方的以下 Ra
这是我的方法 void login(Event event);我想知道 Kotlin 中应该如何 最佳答案 在 Kotlin 中通配符运算符是 * 。它指示编译器它是未知的,但一旦知道,就不会有其他类
看下面的代码 for story in book if story.title.length < 140 - var story
我正在尝试用 C 语言学习字符串处理。我写了一个程序,它存储了一些音乐轨道,并帮助用户检查他/她想到的歌曲是否存在于存储的轨道中。这是通过要求用户输入一串字符来完成的。然后程序使用 strstr()
我正在学习 sscanf 并遇到如下格式字符串: sscanf("%[^:]:%[^*=]%*[*=]%n",a,b,&c); 我理解 %[^:] 部分意味着扫描直到遇到 ':' 并将其分配给 a。:
def char_check(x,y): if (str(x) in y or x.find(y) > -1) or (str(y) in x or y.find(x) > -1):
我有一种情况,我想将文本文件中的现有行包含到一个新 block 中。 line 1 line 2 line in block line 3 line 4 应该变成 line 1 line 2 line
我有一个新项目,我正在尝试设置 Django 调试工具栏。首先,我尝试了快速设置,它只涉及将 'debug_toolbar' 添加到我的已安装应用程序列表中。有了这个,当我转到我的根 URL 时,调试
在 Matlab 中,如果我有一个函数 f,例如签名是 f(a,b,c),我可以创建一个只有一个变量 b 的函数,它将使用固定的 a=a1 和 c=c1 调用 f: g = @(b) f(a1, b,
我不明白为什么 ForEach 中的元素之间有多余的垂直间距在 VStack 里面在 ScrollView 里面使用 GeometryReader 时渲染自定义水平分隔线。 Scrol
我想知道,是否有关于何时使用 session 和 cookie 的指南或最佳实践? 什么应该和什么不应该存储在其中?谢谢! 最佳答案 这些文档很好地了解了 session cookie 的安全问题以及
我在 scipy/numpy 中有一个 Nx3 矩阵,我想用它制作一个 3 维条形图,其中 X 轴和 Y 轴由矩阵的第一列和第二列的值、高度确定每个条形的 是矩阵中的第三列,条形的数量由 N 确定。
假设我用两种不同的方式初始化信号量 sem_init(&randomsem,0,1) sem_init(&randomsem,0,0) 现在, sem_wait(&randomsem) 在这两种情况下
我怀疑该值如何存储在“WORD”中,因为 PStr 包含实际输出。? 既然Pstr中存储的是小写到大写的字母,那么在printf中如何将其给出为“WORD”。有人可以吗?解释一下? #include
我有一个 3x3 数组: var my_array = [[0,1,2], [3,4,5], [6,7,8]]; 并想获得它的第一个 2
我意识到您可以使用如下方式轻松检查焦点: var hasFocus = true; $(window).blur(function(){ hasFocus = false; }); $(win
我是一名优秀的程序员,十分优秀!