0

我不明白为什么下面的代码会失败:

char    toto[54], titi[54]; //I already tried with allocations

sscanf_s(line, "%s-%s", &toto, &titi, sizeof(toto) + sizeof(titi));

或者

sscanf_s(line, "%s-%s", &toto, &titi, sizeof(toto) + sizeof(titi));

我的问题只是字符串(float、int、double 等都可以),我使用 Visual 2010。

有人有想法吗?非常感谢您的回答。

4

2 回答 2

5

sscanf_s()参考页面:

sscanf_s 函数将数据从缓冲区读取到每个参数给定的位置。格式字符串后面的参数指定指向变量的指针,该变量的类型对应于格式中的类型说明符。与不太安全的版本 sscanf 不同,在使用类型字段字符 c、C、s、S 和 [ 时需要缓冲区大小参数。必须在每个需要它的缓冲区之后作为附加参数提供以字符为单位的缓冲区大小。有关详细信息,请参阅 scanf_s、_scanf_s_l、wscanf_s、_wscanf_s_l 和 scanf 类型字段字符。

这意味着每个缓冲区后面必须跟它的大小:

sscanf_s(line, "%s-%s", toto, sizeof(toto), titi, sizeof(titi));

此外,-is not 空格字符不会充当"%s"格式说明符的终止符,因此如果line包含hello-world,则它将被读入toto并且titi不会被分配。要-用作终止符,请使用扫描集:

if (2 == sscanf_s(line, "%[^-]-%s", toto, sizeof(toto), titi, sizeof(titi)))
{
    /* Both 'toto' and 'titi' where assigned. */
}
于 2012-11-07T22:20:43.130 回答
0

除非sscanf_s()很奇怪,否则您不想传递指向数组 ( &toto) 的指针,而是传递指向数组第一个元素( ) 的指针toto。此外,安全函数的目标是避免缓冲区溢出,即,您需要为每个单独的缓冲区指定它包含多少个字符:

if (sscanf_s(line, "%s-%s", toto, sizeof(toto), titi, sizeof(titi)) == 2) {
    ...
}

(错误处理,因为尝试读取应始终在处理数据之前检查是否发生错误)

就个人而言,我会使用std::istringstream字符串和合适的自定义操纵器并完全避免这个问题:

std::istream& minus(std::istream& in) {
    if ((in >> std::ws).peek() != std::char_traits<char>::to_int_type('-')) {
        in.setstate(std::ios_base::failbit);
    }
    else {
        in.ignore();
    }
    return in;
}

std::string toto, titi;
if (std::istringstream(line) >> std::skipws >> toto >> minus >> titi)) {
    ...
}
于 2012-11-07T22:27:03.047 回答