首页 > 解决方案 > ctrl-d, ctrl-c 是如何被 C/scanf 处理的

问题描述

让我们采取以下程序:

# include<stdio.h>
int main(void)
{
    int status, current_number, sum=0;
    printf("Enter a number: ");
    while(status=scanf("%d", &current_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)被识别为-1CtrlC只是导致程序退出。这两个转义序列通常在 C 中如何处理?为什么要区别scanf对待?ctrl-cctrl-d

标签: cescaping

解决方案


scanf函数不会任何特殊方式处理这些字符,它甚至看不到这些字符。发生的情况是终端驱动程序(至少在类 UNIX 系统(a)下)截获这些击键并将它们转换为特殊操作。

对于CTRL-d,它会关闭标准输入文件,以便读取它的任何代码都会得到EOF- 这就是-1您所看到的(表示读取时某些描述的错误)。

对于CTRL-c,它会SIGINT发出信号,如果未被捕获,它将终止您的程序。

请记住,这些是这些操作的默认键绑定,可以更改它们stty以使用不同的键绑定。默认值 (intreof) 如下所示 (^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", &current_number)) {

循环只会在scanf返回零时退出,如果无法扫描整数(例如通过输入非数字XYZZY),就会发生这种情况。对于任何非零值,它将继续,包括您返回-1错误/文件结尾。

更好的循环是:

while((status = scanf("%d", &current_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: ", curr, 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)识别为文件结束指示符。


推荐阅读