防more命令

more 是显示文件第一屏的内容,在屏幕的底部, more 用反白字体显示文件的百分比,这时如果按空格键,文件的下一屏内容会显示出来,如果按回车键,显示的则是下一行,如果输入"q" ,结束显示,如果输入"h" ,显示出来的是more 的联机帮助。注意,当按空格键或输入"q"后,程序会立即响应,而无需再按回车键。
more 有3 种用法:

$ more filename
$ command | more
$ more < filename

第一种情况, more 显示文件filename 的内容;
第二种情况, more 将command 命令的输出分页显示;
第三种情况, more 从标准输入获取要分页显示的内容,而这时more 的标准输入被重定向到文件filename 。
more 的工作流程如下:

  1. 从输入流显示24行文字
  2. 打印 [more?] 信息
  3. 输入 Enter , SPACE, or q
  4. 如果输入 Enter, 下一行
  5. 输入 q 退出

在linux 系统中增加新的命令是一件很容易的事。把程序的可执行文件放到以下任意一个目录就可以了: / bin 、/ u sr / bin 、/ usr/ loca l/ bin ,这些目录里面存放着很多系统命令。
接下来要编写的程序应该像实际的more 一样,有足够的灵活性,也就是说,如果在命令行中给出了文件名,那么就分页显示这个文件,否则的话,从标准输入得到要分页显示的内容。
在主函数中判断应该从文件还是标准输入中获取数据,并打开相应的数据源,然后调用do_more 函数, do_more 将数据显示在显示器上,满一屏后,调用see_more 函数接收用户的输入,以决定下一步的动作。
/dev/tty设备文件,这是键盘和显示器的设备描述文件,向这个文件写相当于显示在用户的屏幕上,读相当于从键盘获取用户的输入。即使程序的输入/输出被" < "或">"重定向,程序还是可以通过这个文件与终端交换数据。

#include 
#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;
}

你可能感兴趣的:(防more命令)