Linux系统下解决getch()输入数值不回显示问题


    在linux系统下开发C 程序却会遇到系统不支持conio.h头文件,无法使用getch()不回显函数。下面就演示如何构建函数实现数值输入不回显。
    
#include <stdio.h> 
#include <termios.h> 
#include <unistd.h> 
#include <errno.h> 
#define ECHOFLAGS (ECHO | ECHOE | ECHOK | ECHONL) 
//函数set_disp_mode用于控制是否开启输入回显功能 
//如果option为0,则关闭回显,为1则打开回显 
int set_disp_mode(int fd,int option) 
{ 
  int err; 
  struct termios term; 
  if(tcgetattr(fd,&term)==-1){ 
   perror("Cannot get the attribution of the terminal"); 
   return 1; 
  } 
  if(option) 
    term.c_lflag|=ECHOFLAGS; 
  else 
    term.c_lflag &=~ECHOFLAGS; 
  err=tcsetattr(fd,TCSAFLUSH,&term); 
  if(err==-1 && err==EINTR){ 
    perror("Cannot set the attribution of the terminal"); 
    return 1; 
  } 
  return 0; 
} 
//函数getpasswd用于获得用户输入的密码,并将其存储在指定的字符数组中 
int getpasswd(char* passwd, int size) 
{ 
  int c; 
  int n = 0; 
  printf("Please Input password:"); 
  do{ 
   c=getchar(); 
   if (c != '\n'|c!='\r'){ 
     passwd[n++] = c; 
   } 
  }while(c != '\n' && c !='\r' && n < (size - 1)); 
  passwd[n] = '\0'; 
  return n; 
} 
int main() 
{ 
  char *p,passwd[20],name[20]; 
  printf("Please Input name:"); 
  scanf("%s",name); 
  getchar();//将回车符屏蔽掉 
  //首先关闭输出回显,这样输入密码时就不会显示输入的字符信息 
  set_disp_mode(STDIN_FILENO,0); 
  //调用getpasswd函数获得用户输入的密码 
  getpasswd(passwd, sizeof(passwd));  
  p=passwd; 
  while(*p!='\n') 
   p++; 
  *p='\0'; 
  printf("\nYour name is: %s",name); 
  printf("\nYour passwd is: %s\n", passwd); 
  printf("Press any key continue ...\n"); 
  set_disp_mode(STDIN_FILENO,1); 
  getchar(); 
  return 0; 
}

    运行结果:
    
    说明:Linux下C编程遇到要输入密码的问题,可输入的时候密码总不能让人看见吧,本来想用getch()来解决输入密码无回显的问题的,不料Linux-C中不支持getch(),我也没有找到功能类似的函数代替,上面这个例子达到了预期的效果。
    PS:linux getch()实现代码如下所示:
    
#include <termio.h>
int getch(void)
{
   struct termios tm, tm_old;
   int fd = 0, ch;
   if (tcgetattr(fd, &tm) < 0) {//保存现在的终端设置
     return -1;
   }
   tm_old = tm;
   cfmakeraw(&tm);//更改终端设置为原始模式,该模式下所有的输入数据以字节为单位被处理
   if (tcsetattr(fd, TCSANOW, &tm) < 0) {//设置上更改之后的设置
     return -1;
   }
   ch = getchar();
   if (tcsetattr(fd, TCSANOW, &tm_old) < 0) {//更改设置为最初的样子
     return -1;
   }
   return ch;
}