设计模式之6大原则(6)开闭原则

1. more第一版

实现基础功能,显示每一页固定24行文本,“q Enter”退出, “Enter” 下一行, “space Enter”下一页。

/*************************************************************************
  > File Name: more01.c
  > Author: qianlv
  > Mail: [email protected]
  > Created Time: 2014年04月24日 星期四 14时58分25秒
  > more01.c - version 0.1 of more
  > read and print 24 lines then pause for a few special commands
 ************************************************************************/

#include<stdio.h>
#include <stdlib.h>
#define PAGELEN 24
#define LINELEN 514
void do_more(FILE *);
int see_more();
int main(int ac, char *av[])
{
    FILE *fp;
    if(ac==1)
      do_more(stdin);// more 后无参数,读取stdin
    else
    {
        while(--ac)
          if( (fp = fopen(* ++av, "r")) != NULL) // 打开文件
          {
              printf("%s\n", *av);
              do_more(fp);
              fclose(fp);
          }
          else
            exit(1);
    }
    return 0;
}

void do_more(FILE *fp)
{
    char line[LINELEN];
    //int see_more();
    int reply;
    int number_line = 0;
    while(fgets(line, LINELEN, fp) != NULL)
    {
        if(number_line == PAGELEN)
        {
            reply = see_more();
            if(reply == 0)
              break;
            number_line -= reply;
        }
        if( fputs(line, stdout) == EOF)
          exit(1);
        number_line ++;
    }
}

int see_more()
{
    int c;
    printf("\033[7m more? \033[m");
    while( (c = getchar()) != EOF )
    {
        if(c == ‘q‘)
          return 0;
        if(c == ‘ ‘)
          return PAGELEN;
        if(c == ‘\n‘)
          return 1;
    }
    return 0;
}

2.more第二版

解决上一个版本“ls -l /etc |  ./more01”, “ls -l /etc” 输出重定向为“./more01”  输入时 由于see_more() 函数中getchar()与do_more(FILE *fp)中读取都是stdin中的数据,时输出一页后不回暂停等待命令。

解决方法是: see_more()改为通过/dev/tty(键盘与显示设备的描述文件),读取键。

/*************************************************************************
	> File Name: more02.c
	> Author: qianlv
	> Mail: [email protected]
	> Created Time: 2014年04月24日 星期四 15时39分51秒
    > more02.c - version 0.2 of more
    > read and print 24 lines the pause for a few special commands
    > feature of version 0.2: reads form /dev/tty for commands
 ************************************************************************/

#include<stdio.h>
#include <stdlib.h>
#define PAGELEN 24
#define LINELEN 514
void do_more(FILE *);
int see_more(FILE *);
int main(int ac, char *av[])
{
    FILE *fp;
    if(ac==1)
      do_more(stdin);
    else
    {
        while(--ac)
          if( (fp = fopen(* ++av, "r")) != NULL)
          {
              do_more(fp);
              fclose(fp);
          }
          else
            exit(1);
    }
    return 0;
}

void do_more(FILE *fp)
{
    char line[LINELEN];
    //int see_more();
    int reply;
    int number_line = 0;
    FILE *fp_tty;
    fp_tty = fopen("/dev/tty", "r");//打开/dev/tty设备文件
    if(fp_tty == NULL)
      exit(1);
    while(fgets(line, LINELEN, fp) != NULL)
    {
        if(number_line == PAGELEN)
        {
            reply = see_more(fp_tty);
            if(reply == 0)
              break;
            number_line -= reply;
        }
        if( fputs(line, stdout) == EOF)
          exit(1);
        number_line ++;
    }
}

int see_more(FILE *cmd)
{
    int c;
    printf("\033[7m more? \033[m");
    while( (c = getc(cmd)) != EOF ) //此处的getchar()从stdin读取数据,getc(cmd)从文件cmd(/dev/tty)中读入数据
    {
        if(c == ‘q‘)
          return 0;
        if(c == ‘ ‘)
          return PAGELEN;
        if(c == ‘\n‘)
          return 1;
    }
    return 0;
}

3. more第三版

通过修改终端属性,无需输入回车,立即响应输入字符命令

/*************************************************************************
	> File Name: more04.c
	> Author: qianlv
	> Mail: [email protected]
	> Created Time: 2014年04月25日 星期五 10时23分22秒
        > 添加键入字符立即响应程序
 ************************************************************************/

#include<stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <termios.h>
#define PAGELEN 24
#define LINELEN 514
void do_more(FILE *);
int see_more(FILE *);
int main(int ac, char *av[])
{
    FILE *fp;
    if(ac==1)
      do_more(stdin);
    else
    {
        while(--ac)
          if( (fp = fopen(* ++av, "r")) != NULL)
          {
              do_more(fp);
              fclose(fp);
          }
          else
            exit(1);
    }
    return 0;
}

void do_more(FILE *fp)
{
    char line[LINELEN];
    int reply;
    int number_line = 0;
    FILE *fp_tty_in, *fp_tty_out;
    fp_tty_in = fopen("/dev/tty", "r");
    fp_tty_out = fopen("/dev/tty", "w");
    struct termios initial_settings, new_settings;
    tcgetattr(fileno(fp_tty_in), &initial_settings);//获取当前终端的属性。
    new_settings = initial_settings;
    new_settings.c_lflag &= ~ICANON;//设置终端为非标准模式
    //new_settings.c_lflag &= ~ECHO; //设置终端不回显
    //设置读入一个字符,立即返回字符。
    new_settings.c_cc[VMIN] = 1;
    new_settings.c_cc[VTIME] = 0;
    if(tcsetattr(fileno(fp_tty_in), TCSANOW, &new_settings) != 0) { // 重新配置终端接口
        fprintf(stderr, "could not set attributes\n");
    }
    while(fgets(line, LINELEN, fp) != NULL)
    {
        if(number_line == PAGELEN)
        {

            reply = see_more(fp_tty_in);
            if(reply == 0)
              break;
            number_line -= reply;
        }
        if( fputs(line, fp_tty_out) == EOF)
        {
            tcsetattr(fileno(fp_tty_in), TCSANOW, &initial_settings); // 恢复终端接口的配置
             exit(1);
        }
        number_line ++;
    }
    tcsetattr(fileno(fp_tty_in), TCSANOW, &initial_settings);// 恢复终端接口的配置
}
int see_more(FILE *cmd)
{
    int c;
    printf("\033[7m more? \033[m");
    do {
        c = fgetc(cmd);
        if(c == ‘q‘)
          return 0;
        if(c == ‘ ‘)
          return PAGELEN;
        if(c == ‘\n‘)
          return 1;
    }while(1);
    return 0;
}

4. more第四版

解决"more?"重复出现的问题,已经每页行数根据终端大小动态决定,由于每次读取一行文本不等于终端行数,所以存在bug。显示文件已经显示占总文件的百分比,显示多个文件时,分别显示出文件名。

/*************************************************************************
  > File Name: more04.c
  > Author: qianlv
  > Mail: [email protected]
  > Created Time: 2014年04月25日 星期五 14时31分07秒
  > 解决"more?"重复出现的问题,已经每页行数根据终端大小动态决定,由于
  > 文件的一行可能占2行以上终端行数,所有有小bug,显示出已经显示的文
  > 件占文件总大小的百分比,如果显示的是多个文件,那么显示出当前读取
  > 的文件的文件名。
 ************************************************************************/

#include<stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <termios.h>
#include <curses.h>
#include <term.h>
#include <string.h>
#include <sys/stat.h>
#define LINELEN 514
static unsigned long filesize = 0; // 文件的总字节数
static unsigned long input_filesize = 0; // 已经显示的字节数
static FILE *out_stream = (FILE *) 0;
static int filesum = 0; // 当前要显示的文件个数
void clear_more(int, int, FILE *);   //“el”删除一行到行尾,用于清除“more?”.
int cols_more(FILE *fp);  // 获取终端列数
int lines_more(FILE *);  // 获取终端行数
int char_to_terminal(int ); // 一个与putchar函数有相同的参数和返回值的函数,用于tputs函数的调用。
void do_more(FILE *, char *filename);
int see_more(FILE *,int, int);
unsigned long get_filesize(FILE *);
int main(int ac, char *av[])
{
    FILE *fp;
    filesum = ac - 1;
    if(ac==1)
      do_more(stdin,(char *)0);
    else
    {
        while(--ac)
          if( (fp = fopen(* ++av, "r")) != NULL)
          {
              filesize = input_filesize = 0; //清空前一个文件的大小。
              filesize = get_filesize(fp);
              do_more(fp, *av);
              fclose(fp);
          }
          else
            exit(1);
    }
    return 0;
}

void do_more(FILE *fp, char *filename)
{
    char line[LINELEN];
    int reply;
    int number_line = 0;
    FILE *fp_tty_in, *fp_tty_out;
    fp_tty_in = fopen("/dev/tty", "r");
    fp_tty_out = fopen("/dev/tty", "w");
    struct termios initial_settings, new_settings;
    tcgetattr(fileno(fp_tty_in), &initial_settings);//获取当前终端的属性。
    new_settings = initial_settings;
    new_settings.c_lflag &= ~ICANON;//设置终端为非标准模式
    new_settings.c_lflag &= ~ECHO; //设置终端不回显
    //设置读入一个字符,立即返回字符。
    new_settings.c_cc[VMIN] = 1;
    new_settings.c_cc[VTIME] = 0;
    if(tcsetattr(fileno(fp_tty_in), TCSANOW, &new_settings) != 0) { // 重新配置终端接口
        fprintf(stderr, "could not set attributes\n");
    }
    int Pagelen = lines_more(fp_tty_in) - 1; //终端行数
    int PageCol = cols_more(fp_tty_in);  //终端列数
    int add_line;
    if(filesum > 1) //显示的文件个数 > 1 那么把文件名也显示出来。
    {
        fprintf(fp_tty_out, "-------> %s <-------\n",filename);
        number_line = 1;
    }
    while(fgets(line, LINELEN, fp) != NULL)
    {
        if(number_line >= Pagelen) //输出的行数大于终端行数时,即为一页,原因是每次读取文件的一行,可能占用终端2行以上。
        {

            reply = see_more(fp_tty_in,Pagelen, add_line);
            int prePage = Pagelen;
            Pagelen = lines_more(fp_tty_in) - 1;  //终端行数
            PageCol = cols_more(fp_tty_in);   //终端列数
            if(prePage < Pagelen)
                clear_more(Pagelen-1, 0, fp_tty_out);   //移动游标至"more?"这一行最前面,然后清除此行。
            else
                clear_more(Pagelen, 0, fp_tty_out);
            if(reply == 0)
              break;
            if(number_line != Pagelen && reply == 1)  // 当终端变大时,且按下时回车“enter”,应把number_line改为终端倒数第二行。
                number_line = Pagelen -1;
            else
                number_line -= reply;

        }
        if( fputs(line, fp_tty_out) == EOF)
        {
            tcsetattr(fileno(fp_tty_in), TCSANOW, &initial_settings); // 恢复终端接口的配置
            exit(1);
        }
        int line_len = strlen(line);
        input_filesize += (unsigned long)line_len;//叠加字节个数。
        add_line = (line_len + PageCol - 1)/PageCol;
        number_line += add_line; //文档的行数不等于终端显示的行数,因为一行字符串可能占据2行以上。
        //number_line ++;
        //fprintf(te, "%d: %d - %d %s",nll++,number_line, add_line,line);
    }
    tcsetattr(fileno(fp_tty_in), TCSANOW, &initial_settings);// 恢复终端接口的配置
}
void clear_more(int posx,int posy,FILE *fp)
{
    char *el;
    char *cursor;
    out_stream = fp;
    cursor = tigetstr("cup");
    el = tigetstr("el");
    tputs(tparm(cursor, posx, posy), 1, char_to_terminal); //移动关闭至(posx,posy)处。
   //////////////// sleep(1);
    tputs(el, 1,  char_to_terminal);//清除此行至行尾。

}
int see_more(FILE *cmd,int Pagelen, int add_line)
{
    int c;
    if(filesize > 0 ) // 如果重定向的输入无法获取大小,则不要显示百分比。
        printf("\033[7m more? \033[m %lu%%",input_filesize*100/filesize);
    else
        printf("\033[7m more? \033[m ");

    do {
        c = fgetc(cmd);
        if(c == ‘q‘)
          return 0;
        if(c == ‘ ‘)
        {
            return Pagelen;
        }
        if(c == ‘\n‘ || c == ‘\r‘) //非标准模式下,默认回车和换行之间的映射已不存在,所以检查回车符‘\r‘。
          return add_line;
    }while(1);
    return 0;
}
int char_to_terminal(int char_to_write)
{
    if(out_stream) putc(char_to_write,out_stream);
    return 0;
}
int lines_more(FILE *fp)
{
    int nrows;
    setupterm(NULL, fileno(fp), (int *)0);
    nrows = tigetnum("lines");
    return nrows;
}
int cols_more(FILE *fp)
{
    int ncols;
    setupterm(NULL, fileno(fp), (int *)0);
    ncols = tigetnum("cols");
    return ncols;
}
unsigned long get_filesize(FILE *fp)
{
    struct stat buf;
    if(fstat(fileno(fp), &buf) < 0)
        return (unsigned long) 0;
    return (unsigned long) buf.st_size;
}

5. 参考书籍

  1. 《Linux 程序设计 第四版》 by Neil Matthew,Richard Stones 人民邮电出版社。
  2. Unix/Linux 编程实践教程  by Bruce Molay 清华大学出版社。

设计模式之6大原则(6)开闭原则,码迷,mamicode.com

时间: 2024-10-14 05:44:22

设计模式之6大原则(6)开闭原则的相关文章

设计模式七大原则之开闭原则学习

这是在我大学学习过程中,老师给我讲的第一个设计原则:开闭原则,至今为止,我只记住了一句话:程序对修改关闭,对扩展开放.接下来得好好理解一下这句话 一.开闭原则 开闭原则是编程中最基础.最重要的设计原则 基本介绍: (1)一个软件实体如类,模块和函数应该对扩展开放(对于提供方来说),对修改关闭(对于使用方来说).用抽象构建框架,用实现扩展细节. (2)当软件需要变化时,尽量通过扩展软件实体的行为来实现变化,而不是通过修改已有的代码来实现变化. (3)编程中遵循其它原则,以及使用设计模式的目的就是遵

面向对象设计原则之开闭原则

http://blog.csdn.net/lovelion/article/details/7537584 为了满足开闭原则,需要对系统进行抽象化设计,抽象化是开闭原则的关键.在Java.C#等编程语言中,可以为系统定义一个相对稳定的抽象层,而将不同的实现行为移至具体的实现层中完成.在很多面向对象编程语言中都提供了接口.抽象类等机制,可以通过它们定义系统的抽象层,再通过具体类来进行扩展.如果需要修改系统的行为,无须对抽象层进行任何改动,只需要增加新的具体类来实现新的业务功能即可,实现在不修改已有

面向对象原则之一 开放封闭原则(开闭原则)

原文:面向对象原则之一 开放封闭原则(开闭原则) 前言 面向对象有人分为五大原则,分别为单一职责原则.开放封闭原则.依赖倒置原则.接口隔离原则.里氏替换原则. 也有人分为六大原则,分别为单一职责原则.开放封闭原则.依赖倒置原则.接口隔离原则.里氏替换原则.迪米特法则. 现在我们来介绍开放封闭原则,也叫开闭原则 开闭原则 1)概念 官方说法是 软件实体(模块.类.函数等)应该可以扩展,但是不可以修改.也就是说软件对扩展开放,对修改关闭. 需要说明的是,对修改关闭不是说软件设计不能做修改,只是尽量不

设计原则之开闭原则Open Close Principle

翻译自http://www.oodesign.com 设计原则之开闭原则 动机:一个聪明的应用设计和代码编写应该考虑到开发过程中的频繁修改代码.通常情况下,一个新功能的增加会带来很多的修改.这些修改已存在的代码应该要最小化, 总结:软件应该对扩展开发,对修改关闭.装饰器模式,观察者模式,工厂模式可以帮助我们队代码做最小的修改. Bad Example: 缺点: 1.当新的shape被添加,开发者要花大量时间去理解GraphicEditor源码.. 2.添加新shape也许会影响已经存在的功能 /

6大设计原则之开闭原则

开闭原则的定义 开闭原则的定义: 一个软件实体,如类.模块和函数应该对扩展开放,对修改关闭.即一个软件实体应该通过扩展来实现变化,而不是通过修改已有的代码来实现变化. 软件实体包括一下部分 项目或软件产品中按照一定的逻辑规则划分的模块 抽象和类 方法 注意: 开闭原则对扩展开放,对修改关闭,并不意味着不做任何修改,底层模块的变更,必然要有高层模块进行耦合,否则就是一个孤立无意义的代码片段 开闭原则的好处 开闭原则是一个最基本的原则,另外五个原则都是开闭原则的具体形态,是指导设计的工具和方法,而开

设计模式——六大原则之开闭原则(一)

开闭原则的定义 开闭原则(Open Closed Principle,OCP)由勃兰特·梅耶(Bertrand Meyer)提出,他在 1988 年的著作<面向对象软件构造>(Object Oriented Software Construction)中提出:软件实体应当对扩展开放,对修改关闭(Software entities should be open for extension,but closed for modification),这就是开闭原则的经典定义. 这里的软件实体包括以下

设计模式原则之开闭原则

开闭原则(OCP)是面向对象设计中“可复用设计”的基石,是面向对象设计中最重要的原则之一,其它很多的设计原则都是实现开闭原则的一种手段. 遵循开闭原则设计出的模块具有两个主要特征: (1)对于扩展是开放的(Open for extension).这意味着模块的行为是可以扩展的.当应用的需求改变时,我们可以对模块进行扩展,使其具有满足那些改变的新行为.也就是说,我们可以改变模块的功能. (2)对于修改是关闭的(Closed for modification).对模块行为进行扩展时,不必改动模块的源

架构中的设计原则之开闭原则(OCP) - 《java开发技术-在架构中体验设计模式和算法之美》

开闭原则OCP(Open for Extension,Closed for Modification).开闭原则的核心思想是:一个对象对扩张开放,对修改关闭. 其实开闭原则的意思就是:对类的改动是通过增加代码进行的,而不是改动现有的代码.也就是说,软件开发人员一旦写出了可以运行的代码,就不应该去改变它,而是要保证它能一直运行下去,如何才能做到这一点呢?这就需要借助于抽象和多态,即把可能变化的内容抽象出来,从而使抽象的部分是相对稳定,而具体的实现层是可以改变和扩展的. 根据开闭原则,我们改变一个软

设计模式之禅--六大原则之开闭原则(精神领袖)

"对修改关闭,对拓展开放". 一个书店卖书的例子,见代码 public interface IBook { public String getName(); public int getPrice(); public String getAuthor(); } ---------- package com.sdkd.hms; public class NovelBook implements IBook { private String name; private int price;