while(c = getch()) 为什么可以这样写? (C/C++) (C) (OS) (Linux)

凡是C语言的初学者,刚谈到简单的IO时,都会学到以下的程序,只要使用者输入什么,就印什么。

1 #include  < stdio.h >
2
3 int  main()  {
4  char c;
5  while(c = getch()) {
6    printf("%c\n",c);
7  }

8}


当我第一次看到这种写法时,非常的震惊,马上改写成C#测试,当然连compile都不会过,为什么C语言可以这样写呢?若将以上程序改写一般语言的版本,就相当于如下的写法

 1 #include  < stdio.h >
 2
 3 int  main()  {
 4  char c;
 5
 6  while(1{
 7    c = getch();
 8
 9    if (c > 0{
10      printf("%c\n",c);
11    }

12    else {
13      break;
14    }

15  }

16}


如此就一目暸然了,可见C语言真的是语法很精简的语言,回到第一个程序,为什么while(c = getch())这种写法C语言接受呢?char c先得到getch()所传回的char型别,但在C语言,char型别和int是互通的,随即转为该字符所代表的ASCII值,任何一个字符的ASCII值均大于0,而在C语言『非0为true』的哲学下,即可进入while循环内,所以只要在键盘上敲入任何值,皆可随即印出来。在Visual C++,仍可接受『非0为true』的写法,但在g++,会出现warning: suggest parentheses around assignment used as truth value,也就是说,已经不建议『非0为true』的观念,而是要真的在while()中填入true或false,C#则是强烈要求如while(),if()在()中的判断式,必须一定要是true或false才行。不过C算是网开一面,只要加上()或手动将int转成bool即可,也就是改成while((c = getch()))或while((bool)(c = getch())) 即可,当加上()时, compiler还是会将()内的值依照『非0为true』转成ture或false。

完整的程序改法如下

1 #include  < stdio.h >
2
3 int  main()  {
4  char c;
5
6  while((c = getch())) {
7    printf("%c",c);
8  }

9}


以上的程序,无论拿到Visual C++或g++都无法compile,原因式getch()已经从<stdio.h>拔除了,只有在古老的compiler如Turbo C还能用,若在Visual C++,必须改成以下的写法。

 1 /* 
 2(C) OOMusou 2006 http://oomusou.cnblogs.com
 3
 4Filename    : _getch.cpp
 5Compiler    : Visual C++ 8.0
 6Description : Demo how to use _getch() & convert () to bool
 7Release     : 11/28/2006
 8*/

 9
10 /* 
11(C) OOMusou 2006 http://oomusou.cnblogs.com
12
13Filename    : conio__getch.cpp
14Compiler    : Visual C++ 8.0
15Description : Demo how to use getch() in conio.h & convert 
16
17() to bool
18Release     : 11/28/2006
19*/

20 #include  < stdio.h >
21 #include  < conio.h >
22
23 int  main()  {
24  char c;
25
26  while((c = _getch())) {
27    printf("%c",c);
28  }

29}


直于g++,<stdio.h>也不提供getch()了,必须改用 <curses.h>。

 1 /* 
 2(C) OOMusou 2006 http://oomusou.cnblogs.com
 3
 4Filename    : curses_getch.cpp
 5Compiler    : gcc 4.1.0 on Fedora Core 5
 6Description  : Demo how to use getch() in curses & convert 
 7
 8() to bool
 9Release     : 11/28/2006
10*/

11
12 #include  < curses.h >   //  use ncurses, initscr(), noecho(), getch(), 
13
14 printw(), endwin() 
15 #include  < stdlib.h >   //  exit(), EXIT_SUCCESS 
16
17 int  main()  {
18  // Initialize curse scheme 
19  initscr(); 
20  // no echo in curse
21  noecho(); 
22
23  char c; 
24  while(c = getch()) 
25    printw("%c\n",c); 
26  }

27
28  // end curses scheme 
29  endwin(); 
30  exit(EXIT_SUCCESS); 
31}
 


curses的写法由于要想在Console提供图形接口,所以写法和一般传统的C语言有些差异。

你可能感兴趣的:(linux)