如何解决C/scanf
我们来看看下面的程序:
# include<stdio.h>
int main(void)
{
int status,current_number,sum=0;
printf("Enter a number: ");
while(status=scanf("%d",¤t_number)) {
sum += current_number;
printf("Status: %d. The current sum is: %d. Enter another number: ",status,sum);
}
}
输入数字:2
状态:1.当前总数为:2.输入另一个数字:3
状态: 1. 当前总和为: 5. 输入另一个数字:状态:-1。当前总和为:8.输入另一个数字:^C
似乎 CtrlD (EOF) 被识别为 -1
但 CtrlC jus 导致程序退出。这两个转义序列在 C 中通常如何处理?为什么 scanf
对 ctrl-c
和 ctrl-d
的处理方式不同?
解决方法
scanf
函数不会以任何特殊方式处理这些字符,它甚至不会看到这些字符。发生的情况是终端驱动程序(至少在类 UNIX 系统(a)下)拦截这些击键并将它们转换为特殊操作。
对于 CTRL-d,它会关闭标准输入文件,这样任何读取它的代码都会得到一个 EOF
- 这就是您看到的 -1
(表示一个读取时的一些描述错误)。
对于 CTRL-c,它会发出 SIGINT
信号,如果未被捕获,将终止您的程序。
请记住,这些是这些操作的默认键绑定,可以通过 stty
更改它们以使用不同的键绑定。默认的(intr
和 eof
)如下所示(^C
和 ^D
):
pax> stty -a
speed 38400 baud; rows 37; columns 145; line = 0;
intr = ^C; quit = ^\; erase = ^?; kill = ^U; eof = ^D; eol = <undef>; eol2 = <undef>; swtch = <undef>; start = ^Q; stop = ^S; susp = ^Z;
rprnt = ^R; werase = ^W; lnext = ^V; discard = ^O; min = 1; time = 0;
-parenb -parodd -cmspar cs8 -hupcl -cstopb cread -clocal -crtscts
-ignbrk -brkint -ignpar -parmrk -inpck -istrip -inlcr -igncr icrnl ixon -ixoff -iuclc -ixany -imaxbel -iutf8
opost -olcuc -ocrnl onlcr -onocr -onlret -ofill -ofdel nl0 cr0 tab0 bs0 vt0 ff0
isig icanon iexten echo echoe echok -echonl -noflsh -xcase -tostop -echoprt echoctl echoke -flusho -extproc
请记住,这可能不是您想要的:
while(status=scanf("%d",¤t_number)) {
循环只会在 scanf
返回零时退出,如果无法扫描整数(例如通过输入非数字 XYZZY
),则会发生这种情况。对于任何非零值,将继续,包括您在错误/文件结束时返回的 -1
。
更好的循环是:
while((status = scanf("%d",¤t_number)) == 1) {
事实上,由于循环应该只永远为 1
的状态值运行,所以使用它几乎没有意义(其他用于对发生的事情做出最终决定)。我更喜欢这样的:
#include<stdio.h>
int main(void) {
int stat,curr,sum = 0;
// Prompt and loop while user enters valid numbers.
printf("Enter a number: ");
while ((stat = scanf("%d",&curr)) == 1) {
// Accumulate number to sum,output details and ask for next.
sum += curr;
printf("Entered %d,sum is %d,enter another number: ",sum);
}
// Final status -1 if EOF/error,0 if item couldn't be scanned.
if (stat == -1) {
prinf("\nEnd of file or I/O error.\n");
} else {
prinf("Non-numeric data.\n");
}
}
(a) Windows,从内存中,相比之下,只识别行首的 CTRL-z(后跟 ENTER) 作为文件结束指示符。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。