more 是显示文件第一屏的内容,在屏幕的底部, more 用反白字体显示文件的百分比,这时如果按空格键,文件的下一屏内容会显示出来,如果按回车键,显示的则是下一行,如果输入"q" ,结束显示,如果输入"h" ,显示出来的是more 的联机帮助。注意,当按空格键或输入"q"后,程序会立即响应,而无需再按回车键。
more 有3 种用法:
$ more filename
$ command | more
$ more < filename
第一种情况, more 显示文件filename 的内容;
第二种情况, more 将command 命令的输出分页显示;
第三种情况, more 从标准输入获取要分页显示的内容,而这时more 的标准输入被重定向到文件filename 。
more 的工作流程如下:
- 从输入流显示24行文字
- 打印 [more?] 信息
- 输入 Enter , SPACE, or q
- 如果输入 Enter, 下一行
- 输入 q 退出
在linux 系统中增加新的命令是一件很容易的事。把程序的可执行文件放到以下任意一个目录就可以了: / bin 、/ u sr / bin 、/ usr/ loca l/ bin ,这些目录里面存放着很多系统命令。
接下来要编写的程序应该像实际的more 一样,有足够的灵活性,也就是说,如果在命令行中给出了文件名,那么就分页显示这个文件,否则的话,从标准输入得到要分页显示的内容。
在主函数中判断应该从文件还是标准输入中获取数据,并打开相应的数据源,然后调用do_more 函数, do_more 将数据显示在显示器上,满一屏后,调用see_more 函数接收用户的输入,以决定下一步的动作。
/dev/tty设备文件,这是键盘和显示器的设备描述文件,向这个文件写相当于显示在用户的屏幕上,读相当于从键盘获取用户的输入。即使程序的输入/输出被" < "或">"重定向,程序还是可以通过这个文件与终端交换数据。
#include <stdio.h>
#define PAGELEN 24 /*定义每页显示的行数*/
#define LINELEN 512 /*定义每行显示的字节数*/
void do_more(FILE *);
int see_more(FILE *);
int main(int ac, char *av[])
{
FILE *fp; /*定义文件描述符*/
if(ac == 1) /*判断是否有文件名传入,没有就从标准输入传给do_more*/
{
do_more(stdin);
}else{
while(--ac){ /*循环读取命令行参数*/
if((fp = fopen(*++av, "r")) != NULL) /*判断打开文件是否失败*/
{
do_more(fp); /*把文件描述符传给do_more进行内容显示*/
fclose(fp); /*关闭文件*/
}else{
exit(1); /*错误退出*/
}
return 0;
}
}
}
void do_more(FILE *fp)
{
char line[LINELEN];
int num_of_lines = 0;
int see_more(FILE *), reply;
FILE *fp_tty;
fp_tty = fopen("/dev/tty", "r");
if(fp_tty == NULL) /*判断是否打开失败*/
exit(1);
while(fgets(line, LINELEN, fp)){
if(num_of_lines ==PAGELEN){
reply = see_more(fp_tty);
if(reply == 0)
break;
num_of_lines -= reply; /*重设总数*/
}
if(fputs(line, stdout) == EOF)
exit(1);
num_of_lines++;
}
}
int see_more(FILE *cmd)
{
int c;
printf("\033[7m more? \033[m");
while ((c = getc(cmd)) != EOF)
{
if(c == 'q') /*q 退出*/
return 0;
if(c == ' ') /*' ' 下一页*/
return PAGELEN;
if(c == '\n')
return 1;
}
return 0;
}