zoukankan      html  css  js  c++  java
  • HNCU1741:算法3-2:行编辑程序

    http://hncu.acmclub.com/index.php?app=problem_title&id=111&problem_id=1741

    题目描述

          一个简单的行编辑程序的功能是:接收用户从终端输入的程序或数据,并存入用户的数据区。由于用户在终端上进行输入时,不能保证不出差错,因此,若在编辑程序中,“每接收一个字符即存入用户数据区”的做法显然不是很恰当。较好的做法是,设立一个输入缓冲区,用以接收用户输入的一行字符,然后逐行存入用户数据区。允许用户输入出差错,并在发现有误时可以及时更正。例如,当用户发现刚刚键入的一个字符是错的时,可补进一个退格符“#”,以表示前一个字符无效;如果发现当前键入的行内错误较多或难以补救,则可以键入一个退行符“@”,以表示当前行中的字符均无效。例如假设从终端接收了这样的两行字符:
    whil##ilr#e(s#*s)
        outcha@    putchar(*s=#++);
    则实际有效的是下列两行:
    while(*s)
        putchar(*s++);

          为此,可设这个输入缓冲区为一个栈结构,每当从终端接收了一个字符之后先作如下判别:如果它不是退格符也不是退行符,则将该字符压入栈顶;如果是一个退格符,则从栈顶删去一个字符;如果它是一个退行符,则将字符栈清为空栈。上述处理过程可用下面算法描述之:

    图:行编辑程序算法

    输入格式

    若干行程序或者数据,每行不超过200个字符。

    输出

    经过行编辑程序处理过后的输出。

    样例输入

    whil##ilr#e(s#*s)
    outcha@ putchar(*s=#++);

    样例输出

    while(*s)
    putchar(*s++);

    #include<string.h>
    #include<ctype.h>
    #include<malloc.h> /* malloc()等 */
    #include<limits.h> /* INT_MAX等 */
    #include<stdio.h> /* EOF(=^Z或F6),NULL */
    #include<stdlib.h> /* atoi() */
    #include<math.h> /* floor(),ceil(),abs() */
    /* 函数结果状态代码 */
    #define TRUE 1
    #define FALSE 0
    #define OK 1
    #define ERROR 0
    #define INFEASIBLE -1
    typedef int Status; /* Status是函数的类型,其值是函数结果状态代码,如OK等 */
    typedef int Boolean; /* Boolean是布尔类型,其值是TRUE或FALSE */
    
    #define STACK_INIT_SIZE 10 /* 存储空间初始分配量 */
    #define STACKINCREMENT 2 /* 存储空间分配增量 */
    typedef char SElemType; /* 定义栈元素类型为整型 */
    typedef struct SqStack
    {
        SElemType *base; /* 在栈构造之前和销毁之后,base的值为NULL */
        SElemType *top; /* 栈顶指针 */
        int stacksize; /* 当前已分配的存储空间,以元素为单位 */
    } SqStack; /* 顺序栈 */
    
    Status InitStack(SqStack *S)
    {
        /* 构造一个空栈S */
        (*S).base=(SElemType *)malloc(STACK_INIT_SIZE*sizeof(SElemType));
        if(!(*S).base)
            exit(OVERFLOW); /* 存储分配失败 */
        (*S).top=(*S).base;
        (*S).stacksize=STACK_INIT_SIZE;
        return OK;
    }
    
    Status Push(SqStack *S,SElemType e)
    {
        /* 插入元素e为新的栈顶元素 */
        if((*S).top-(*S).base>=(*S).stacksize) /* 栈满,追加存储空间 */
        {
            (*S).base=(SElemType *)realloc((*S).base,((*S).stacksize+STACKINCREMENT)*sizeof(SElemType));
            if(!(*S).base)
                exit(OVERFLOW); /* 存储分配失败 */
            (*S).top=(*S).base+(*S).stacksize;
            (*S).stacksize+=STACKINCREMENT;
        }
        *((*S).top)++=e;
        return OK;
    }
    
    Status Pop(SqStack *S,SElemType *e)
    {
        /* 若栈不空,则删除S的栈顶元素,用e返回其值,并返回OK;否则返回ERROR */
        if((*S).top==(*S).base)
            return ERROR;
        *e=*--(*S).top;
        return OK;
    }
    
    Status ClearStack(SqStack *S)
    {
        /* 把S置为空栈 */
        (*S).top=(*S).base;
        return OK;
    }
    
    Status DestroyStack(SqStack *S)
    {
        /* 销毁栈S,S不再存在 */
        free((*S).base);
        (*S).base=NULL;
        (*S).top=NULL;
        (*S).stacksize=0;
        return OK;
    }
    
    Status StackTraverse(SqStack S,Status(*visit)(SElemType))
    {
        /* 从栈底到栈顶依次对栈中每个元素调用函数visit()。 */
        /* 一旦visit()失败,则操作失败 */
        while(S.top>S.base)
            visit(*S.base++);
        printf("
    ");
        return OK;
    }
    
    void LineEdit()   // 算法3.2
    {
        //利用字符栈S,从终端接收一行并传送至调用过程的数据区。
        char ch, *temp;
        SqStack S;
        InitStack(&S); //构造空栈S
        ch = getchar(); //从终端接收第一个字符
        while (ch != EOF)   //EOF为全文结束符
        {
            while (ch != EOF && ch != '
    ')
            {
                switch (ch)
                {
                case '#':
                    Pop(&S, &ch);
                    break; // 仅当栈非空时退栈
                case '@':
                    ClearStack(&S);
                    break; // 重置S为空栈
                default:
                    Push(&S, ch);
                    break; // 有效字符进栈,未考虑栈满情形
                }
                ch = getchar(); // 从终端接收下一个字符
            }
            temp = S.base;
            while (temp != S.top)
            {
                printf("%c", *temp);
                ++temp;
            }
            // 将从栈底到栈顶的栈内字符传送至调用过程的数据区;
            ClearStack(&S); // 重置S为空栈
            printf("
    ");
            if(ch != EOF)
            {
                ch = getchar();        // 读取下一行的第一个字符
            }
        }
        DestroyStack(&S);
    }
    
    int main()
    {
        LineEdit();
        return 0;
    }
    


     

  • 相关阅读:
    PHP 将二维数组中某列值作为数组的键名
    MySQL 8下忘密码后重置密码
    单一职责原则
    Linux下安装SVN服务端小白教程
    go 代码玩耍
    centos7 docker开启认证的远程端口2376配置教程
    Dockerfile RUN,CMD,ENTRYPOINT命令区别
    wait-for-it.sh脚本控制docker-compose启动顺序详解
    阿里云服务器漏洞修复_2020.5.22
    Let's Encrypt 免费通配符 SSL 证书申请教程
  • 原文地址:https://www.cnblogs.com/javawebsoa/p/3199113.html
Copyright © 2011-2022 走看看