13.18 为什么大家都说不要使用 scanf()?那我该用什么来代替呢?
scanf() 有很多问题 --- 参见问题 13.15, 13.16 和 13.17。而且, 它的 %s 格式有着和 gets() 一样的问题 (参见问题 13.20) --- 很难保证接收缓冲不溢出。
更一般地讲, scanf() 的设计使用于相对结构化的, 格式整齐的输入。设计上, 它的名称就是来自于 ``scan formatted"。如果你注意到, 它会告诉你成功或 失败, 但它只能提供失败的大略位置, 至于失败的原因, 就无从得知了。 对 scanf() 多得体的错误恢复几乎是不可能的; 通常先用类似 fgets() 的函数 读入整行, 然后再用 sscanf() 或其它技术解释。strtol(), strtok() 和 atoi() 等函数通常有用; 参见问题 14.4。如果你真的要用任何 scanf 的变体, 你要确保检查返回值, 以确定找到了期待的值。而使用 %s 格式的时候, 一定要 小心缓冲区溢出。
参考资料: [K&R2, Sec. 7.4 p. 159]。
scanf() 有很多问题 --- 参见问题 13.15, 13.16 和 13.17。而且, 它的 %s 格式有着和 gets() 一样的问题 (参见问题 13.20) --- 很难保证接收缓冲不溢出。
更一般地讲, scanf() 的设计使用于相对结构化的, 格式整齐的输入。设计上, 它的名称就是来自于 ``scan formatted"。如果你注意到, 它会告诉你成功或 失败, 但它只能提供失败的大略位置, 至于失败的原因, 就无从得知了。 对 scanf() 多得体的错误恢复几乎是不可能的; 通常先用类似 fgets() 的函数 读入整行, 然后再用 sscanf() 或其它技术解释。strtol(), strtok() 和 atoi() 等函数通常有用; 参见问题 14.4。如果你真的要用任何 scanf 的变体, 你要确保检查返回值, 以确定找到了期待的值。而使用 %s 格式的时候, 一定要 小心缓冲区溢出。
参考资料: [K&R2, Sec. 7.4 p. 159]。
授人以渔,不授人以鱼。