终端I/O应用很广泛,用于终端、计算机之间的直接连线、调制解调器以及打印机等等。终端I/O有两种不同的工作模式:

  (1)规范模式输入处理:终端输入以行为单位进行处理,对于每个读要求,终端驱动程序最多返回一行。(默认模式)

  (2)非规范模式输入处理:输入字符并不组成行。

  终端设备是由一般位于内核的终端驱动程序控制的,每个终端设备有一个输入队列和一个输出队列。如下图:

可以检测和更改的终端设备特性都包含在termios结构中。该结构定义在<termios.h>
struct termios{
tcflag_t   c_iflag;    输入标志
tcflag_t   c_oflag;     输出标志
tcflag_t   c_cflag;       控制标志
tcflag_t   c_lflag;       本地标志
cc_t     c_cc[NCCS];    控制字符
}
终端I/O函数

 

写个程序,更改特殊字符,禁用中断字符和更改文件结束符。程序如下:

1 #include <stdio.h>
2 #include <stdlib.h>
3 #include <termios.h>
4 #include <errno.h>
5 #include <unistd.h>
6
7 int main()
8 {
9 struct termios term;
10 long vdisable;
11 //判断标准输入是否是终端设备
12 if(isatty(STDIN_FILENO) == 0)
13 {
14 printf("Standard input is not a terminal device.\n");
15 exit(-1);
16 }
17 if((vdisable = fpathconf(STDIN_FILENO,_PC_VDISABLE))<0)
18 {
19 perror("fpathconf eeror or _POSIX_VDISABLE not in effect");
20 exit(-1);
21 }
22 //获取termios结构
23 if(tcgetattr(STDIN_FILENO,&term) < 0)
24 {
25 perror("tcgetattr error");
26 exit(-1);
27 }
28
29 term.c_cc[VINTR] = vdisable;
30 term.c_cc[VEOF] = 2;
31 //设置termios结构
32 if(tcsetattr(STDIN_FILENO,TCSAFLUSH,&term) < 0)
33 {
34 perror("tcsetattr error");
35 exit(-1);
36 }
37 return 0;
38 }
 

获取和设置终端属性函数:

int tcgetattr(int fd, struct termios *termios_p);
int tcsetattr(int fd, int optional_actions,const struct termios *termios_p);

调用以上函数屏蔽标志取或设置一个值,程序如下:

 1 #include <stdio.h>
2 #include <stdlib.h>
3 #include <termios.h>
4 #include <errno.h>
5 #include <unistd.h>
6
7 int main()
8 {
9 struct termios term;
10 //获取termios结构
11 if(tcgetattr(STDIN_FILENO,&term) < 0)
12 {
13 perror("tcgetattr error");
14 exit(-1);
15 }
16 switch(term.c_cflag & CSIZE)
17 {
18 case CS5:
19 printf("5 bits/byte\n");
20 break;
21 case CS6:
22 printf("6 bits/byte\n");
23 break;
24 case CS7:
25 printf("7 bits/byte\n");
26 break;
27 case CS8:
28 printf("8 bits/byte\n");
29 break;
30 default:
31 printf("Unknown bityes/byte\n");
32 }
33 term.c_cflag &= ~CSIZE; //字符长度清0
34 term.c_cflag |= CS5; //设置为8 bites/byte
35 if(tcsetattr(STDIN_FILENO,TCSANOW,&term) < 0)
36 {
37 perror("tcsetattr error");
38 exit(-1);
39 }
40 return 0;
41 }
 

stty命令:在终端中输入stty -a命令显示终端的所有选项,执行命令结果如下:

终端标识:在大多数UNIXi系统中,控制终端的名字是/dev/tty。

char *ctermid(char *s);  //获取终端控制名字

int isatty(int fd); //判断fd是否为终端设备

char *ttyname(int fd);  // 获取终端设备的路径名

写个程序输出控制终端的标识符信息,程序如下:

1 #include <stdio.h>
2 #include <stdlib.h>
3 #include <termios.h>
4 #include <unistd.h>
5 #include <string.h>
6 static char ctermid_name[L_ctermid];
7 char* my_ctermid(char *str)
8 {
9 if(str == NULL)
10 str = ctermid_name;
11 return (strcpy(str,"/dev/tty"));
12 }
13 int main()
14 {
15 char tername[50];
16 char *name;
17 ctermid(tername);
18 printf("terminate name is: %s\n",tername);
19 my_ctermid(tername);
20 printf("my terminate name is: %s\n",tername);
21 printf("Test isatty() function.\n");
22 printf("fd 0 is: %s\n",isatty(0)? "tty" : "not a tty");
23 printf("fd 1 is: %s\n",isatty(1)? "tty" : "not a tty");
24 printf("fd 2 is: %s\n",isatty(2)? "tty" : "not a tty");
25 printf("Test ttyname() function.\n");
26 if(isatty(0))
27 {
28 name = ttyname(0);
29 if(name == NULL)
30 name ="undefined";
31 }
32 else
33 name = "not a tty";
34 printf("fd 0 :%s\n",name);
35 if(isatty(1))
36 {
37 name = ttyname(1);
38 if(name == NULL)
39 name ="undefined";
40 }
41 else
42 name = "not a tty";
43 printf("fd 1 :%s\n",name);
44 if(isatty(2))
45 {
46 name = ttyname(2);
47 if(name == NULL)
48 name ="undefined";
49 }
50 else
51 name = "not a tty";
52 printf("fd 2 :%s\n",name);
53 exit(0);
54 }
 

程序执行结果如下:

终端的窗口大小:内核为每个终端和伪终端保存了一个窗口大小结构winszie,用ioctl函数的TIOCGWINSZ命令可以获取此结构的当前值。

struct winsize {

  unsigned short ws_row;

  unsigned short ws_col;

  unsigned short ws_xpixel; /* unused */

  unsigned short ws_ypixel; /* unused */

};

写个程序打印终端窗口大小,程序如下:

 1 #include <stdio.h>
2 #include <stdlib.h>
3 #include <termios.h>
4 #include <unistd.h>
5 #include <string.h>
6 #include <sys/ioctl.h>
7 #include <signal.h>
8 #include <errno.h>
9
10 static void pr_winsize(int fd)
11 {
12 struct winsize size;
13 if(ioctl(fd,TIOCGWINSZ,(char *)&size) < 0)
14 {
15 perror("ioctl() error");
16 exit(-1);
17 }
18 printf("%d rows,%d columns\n",size.ws_row,size.ws_col);
19 }
20 static void sig_winch(int signo)
21 {
22 printf("SIGWINCH received\n");
23 pr_winsize(STDIN_FILENO);
24 }
25 int main()
26 {
27 if(isatty(STDIN_FILENO) == 0)
28 {
29 printf("STDIN_FILENO is not terminate device.\n");
30 exit(1);
31 }
32 if(signal(SIGWINCH,sig_winch) == SIG_ERR)
33 {
34 perror("signal() error");
35 exit(-1);
36 }
37 pr_winsize(STDIN_FILENO);
38 for( ; ;)
39 pause();
40 }

程序执行结果如下:

总结:本章介绍了终端,涉及到很多系统底层的知识,很多参数。看的时候只是了解了一些基本的终端操作,还要很多地方不懂,关键是不知道终端用在什么地方,以后用到了需要回头好好学习一下。

Unix环境高级编程(十九)终端I/O的更多相关文章

  1. Unix环境高级编程(十五)高级I/O

    1.非阻塞I/O 对低速设备的I/O操作可能会使进程永久阻塞,这类系统调用主要有如下情况:(1)如果数据并不存在,则读文件可能会使调用者永远阻塞(例如读管道.终端设备和网络设备).(2)如果数据不能立 ...

  2. Unix环境高级编程(十二)线程控制

    本章介绍了一个进程中多个线程之间如何保持数据的似有性及进程的系统调用如何与线程进行交互. 1.线程限制: Single Unix定义了一线线程操作的限制,和其他的限制一样,可以通过sysconf来查询 ...

  3. Unix环境高级编程(十八)高级进程间通信

    本章主要介绍了基于STREAM的管道和UNIX域套接字,这些IPC可以在进程间传送打开文件描述符.服务进程可以使用它们的打开文件描述符与指定的名字相关联,客户进程可以使用这些名字与服务器进程通信. 1 ...

  4. Unix环境高级编程(十四)守护进程实现时间服务器

    守护进程是在后台运行不受终端控制的进程(如输入.输出等),一般的网络服务都是以守护进程的方式运行.守护进程脱离终端的主要原因有两点:(1)用来启动守护进程的终端在启动守护进程之后,需要执行其他任务.( ...

  5. Unix环境高级编程(十)信号续

    1.signal函数 Unix系统的信号机制最简单的接口是signal函数,函数原型如下: #include <signal.h> typedef void (*sighandler_t) ...

  6. Unix环境高级编程(十六)进程间通信

    进程间通信(IPC)是指能在两个进程间进行数据交换的机制.现代OS都对进程有保护机制,因此两个进程不能直接交换数据,必须通过一定机制来完成. IPC的机制的作用: (1)一个软件也能更容易跟第三方软件 ...

  7. (九) 一起学 Unix 环境高级编程 (APUE) 之 线程

    . . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...

  8. (十) 一起学 Unix 环境高级编程 (APUE) 之 线程控制

    . . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...

  9. (十二) 一起学 Unix 环境高级编程 (APUE) 之 进程间通信(IPC)

    . . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...

随机推荐

  1. Android下的Junit测试

    Android SDK 1.5已经将JUnit包含进来了,用过一次,昨天晚上重新用的时候还出了一点问题,还是决定写一篇比较详细的文章,供大家和自己以后使用,写起来也挺方便的,Android下的Juni ...

  2. org.dom4j.DocumentException: unknown protocol: d Nested exception: unknown

    最近用dom4j写了一个修改XML文件的小例子,抛出了如下异常: org.dom4j.DocumentException: unknown protocol: d Nested exception: ...

  3. jacoco统计自动化测试代码覆盖率

    一.下载jacoco, ant工具 jacoco: http://www.eclemma.org/jacoco/: ant: http://ant.apache.org 并添加到环境变量: 二.不停j ...

  4. Java从零开始学二十(集合简介)

    一.为什么需要集合框架 数组的长度是固定的,但是如果写程序时并不知道程序运行时会需要多少对象.或者需要更复杂的方式存储对象,---那么,可以使用JAVA集合框架,来解决这类问题 二.集合框架主要接口 ...

  5. svn log笔记

    背景: svn,版本 1.7.8 以下所有命令直接和svn服务器进行交互并没有checkout代码到本地 1.查询分支从拉出来到现在的变更: svn log branche_url --stop-on ...

  6. ci高级使用方法篇之连接多个数据库

    在我们的项目中有时可能须要连接不止一个数据库.在ci中怎样实现呢? 我们在本地新建了两个数据库,例如以下截图所看到的: 改动配置文件database.php文件为例如以下格式(读者依据自己数据库的情况 ...

  7. 【OpenERP】Report 生成

    以模块oecnj_trainning为例,模块路径: ~/openerp/addons/oecn_training/ ,以下简写为 path/oecn/ Report生成方法:(手写) rml + r ...

  8. libevent个人理解

    1.利用了前置声明来在c语言的基础上进行封装操作.即在include目录下防止event.h等头文件,在这些头文件中只暴露struct的声明却不暴露其定义,对于如event_base等结构的操作均使用 ...

  9. eclipse 图片预览插件

      eclipse 图片预览插件 CreateTime--2018年4月22日22:59:55 Author:Marydon 下载地址:eclipse 图片预览插件 将插件文件夹直接拷贝到eclips ...

  10. 【Oracle】事务处理

    名词解释 DML:Data Manipulation Language (数据库操纵语言) 例如:DELETE.INSERT.UPDATE.SELECT DDL:Data Definition Lan ...