Linux-退格键回显(^H^H^H^H)stty/tcgetattr学习

冷不防 2022-05-20 00:08 389阅读 0赞

这个问题确实挺揪心,退格键(“backspace”)居然是^H^H,不符合我们使用习惯,我们平常使用退格键都习惯删除上一个字符。

可以通过stty来实现或者在程序中tcgetattr+tcsetattr结合实现。

stty:

  1. stu@ubuntu:~/test1$ stty erase ^H
  2. bash下:$ stty erase ^?
  3. 或者把 stty erase ^? 添加到.bash_profile中。
  4. csh下:$ stty erase ^H
  5. 或者把 stty erase ^H 添加到.cshrc

tcgetattr+tcsetattr:

问题描述:

  1. int main(void )
  2. {
  3. //setstty();
  4. char buf[100];
  5. read(STDIN_FILENO,buf,sizeof(buf));
  6. write(STDOUT_FILENO,buf,strlen(buf));
  7. //renew();
  8. return 0;
  9. }
  10. //结果,输入hello(连按几个退格键"backspace",
  11. //出现的效果不是我们想要的,我们希望按下退格键就是删除前一个字符)
  12. /*
  13. stu@ubuntu:~/test1$ ./test
  14. hello^H^H^H^H
  15. hello
  16. stu@ubuntu:~/test1$
  17. */

大概介绍:

  1. int tcgetattr(int fd, struct termios *termios_p);
  2. int tcsetattr(int fd, int optional_actions,
  3. const struct termios *termios_p);
  4. //说明:tcgetattr函数用于获取与终端相关的参数。参数fd为终端的文件描述符,返回的结果保存在termios结构体中,
  5. //tcsetattr函数用于设置终端的相关参数。参数fd为打开的终端文件描述符,
  6. //参数optional_actions用于控制修改起作用的时间,而结构体termios_p中保存了要修改的参数
  7. //termios结构至少包括以下:
  8. tcflag_t c_iflag; //输入模式标志,控制终端输入方式
  9. tcflag_t c_oflag; //输出模式标志,控制终端输出方式
  10. tcflag_t c_cflag; //控制模式标志,指定终端硬件控制信息
  11. tcflag_t c_lflag; //本地模式标志,控制终端编辑功能
  12. cc_t c_cc[NCCS]; //控制字符,用于保存终端驱动程序

tcgetattr/tcsetattr API 学习(man):http://linux.die.net/man/3/tcgetattr

tcgetattr/tcsetattr 前人博客学习:http://blog.chinaunix.net/uid-10747583-id-97303.html

主要解决程序退格键回显问题:

1、获取终端termios信息

2、对应的键功能修改

3、完成后设置生效

(4、程序结束前回复系统的默认状态,根据需要决定)

修改终端控制字符,将终端输入退格键“backspace”,修改为具有真正擦出功能,符合我们日常使用习惯。首先,程序调用tcgetattr函数获得标准 输入的termios信息,将termios结构体中的c_cc[VERASE]控制字符的修改成’\b’;\b代表“backspace”键.而VERASE代表具有擦除.这样的意思就是将擦除功能赋予’\b’键,也就是”backspace”键然后,使用tcsetattr 函数将修改后的termios参数设置到终端中。

具体实现代码如下

  1. #include<stdio.h>
  2. #include<stdlib.h>
  3. #include<string.h>
  4. #include<unistd.h>
  5. #include<termios.h>
  6. #include<errno.h>
  7. struct termios tmp, old;
  8. //设置退格键“backspace”为退格键。
  9. void setstty()
  10. {
  11. //得到系统termion的状态
  12. if(tcgetattr(STDIN_FILENO, &tmp) == -1)
  13. {
  14. printf("tcgetattr error: %s\n", strerror(errno));
  15. return ;
  16. }
  17. old = tmp;
  18. //'\b'为退格键的ASCII码 因此下面这个语句表示退格键被修改为erase功能,即能擦除
  19. tmp.c_cc[VERASE] = '\b';
  20. //设置系统termion的状态,TCSANOW表示设置完成效果立即生效
  21. if(tcsetattr(STDIN_FILENO, TCSANOW, &tmp) == -1)
  22. {
  23. printf("tcsetattr error: %s\n", strerror(errno));
  24. return ;
  25. }
  26. }
  27. //恢复之前系统默认的状态,回显^H
  28. void renew()
  29. {
  30. if(tcsetattr(STDIN_FILENO, TCSANOW, &old) == -1)
  31. {
  32. printf("tcsetattr error: %s\n", strerror(errno));
  33. return ;
  34. }
  35. }
  36. int main(void )
  37. {
  38. setstty();//设置修改并生效
  39. char buf[100];
  40. read(STDIN_FILENO,buf,sizeof(buf));
  41. write(STDOUT_FILENO,buf,strlen(buf));
  42. renew(); //恢复系统的默认状态
  43. return 0;
  44. }

发表评论

表情:
评论列表 (有 0 条评论,389人围观)

还没有评论,来说两句吧...

相关阅读