作业:文件排版(文本文件读写)

【问题描述】

英文电影中参演人员名单一般以某种方式进行排版显示。给定一个未排版的文件listin.txt,该文件中每行参演人员名单由冒号ldquo:rdquo分隔成前后两部分,但格式杂乱无章,单词(由除空格和水平制表符之外的其它字符构成)之间可能有多个空格或水平制表符分隔。编写程序,要求将其按如下排版规则排版输出到另一个文件listout.txt中:

1.从标准输入整数,作为排版后所有各行冒号在一行中的固定位置,输入的整数肯定大于排版后所有各行冒号前的字符个数,位置从1开始计数;

2.冒号左边的单词串以行头为基准左对齐,左边的最后一个单词与冒号之间以空格填充;冒号右边的单词串以冒号基准左对齐,最后一个单词后只有回车换行符,不再有其它字符;

3.冒号左右两边的单词间都只有一个空格分隔,并且要求冒号两边至少各有一个空格。

假设输入文件中每行字符个数不超过100。

【输入形式】

待排版的参演人员名单从当前目录下的listin.txt文件中读入,表示冒号位置的整数从标准输入读入。

【输出形式】

排版后的参演人员名单输出到当前目录下的listout.txt中。

【输入样例】

假设文件listin.txt内容为:

   Digital Intermediate by :   EFILM

Supervising    Digital Colorist : STEVEN J. SCOTT  

 Second Colorist :ANDREW FRANCIS

 Digital Intermediate Producer:LOAN PHAN

Digital  Intermediate Editor:  DEVON MILLER    

表示冒号固定位置的整数为:

40

【输出样例】

文件listout.txt中的内容应为:

【样例说明】

输入的文件listin.txt中有五行参演人员名单,要求排版后冒号位于第40个字符的位置,按照上述排版规则输出到文件listout.txt中。

解决方法

 这道题主要考察两个操作,一是文本文件读写,二是字符串处理。

本题的文件读写只是常规操作,有点难的是字符串处理。

我们按照题目要求,从文本中每读入一行就对这行字符串进行处理。

令人头疼的是,inlist文件中每一行的字符中间可能有连续的空格和换行符,这些都是outlist中不需要的。如何清除这些连续的空格和换行符就是我们要仔细考虑的问题。

我们先去找“:”的位置,把一行分成两个子串进行处理。依次读取字符。如果是换行符,就把它改为空格(不能完全删掉,因为单词之间要保留一个空格);如果既不是冒号也不是换行符,就读入子串。有这样几种情况会使子串在处理过后还有多余空格:空格+空格;空格+换行符;换行符+空格;换行符+换行符。所以我们把这四种情况放到最前面考虑。

这样处理过后,两个子串还是有可能保留多余的空格,但是已经没有换行符了。

这些多余的空格在哪里呢?

就在子串的开头位置。

于是我们只需要清理开头的空格。用迭代器指向子串开始的位置,如果是空格,就删掉它,如果不是,就退出循环。注意,在循环中不需要移动迭代器的位置,因为删掉的是开头的空格,每次删完,迭代器还在开头位置。

(string类的函数很强大,好好利用可以轻松处理字符串。)

#include <iostream>
#include<fstream>
#include<iomanip>
using namespace std;

int main()
{
    int n;
    cin>>n;
    ifstream infile("listin.txt",ios::in);
    ofstream outfile("listout.txt",ios::out);
    string temp;
    if(!infile.is_open())
    {
        cerr<<"Failed!"<<endl;
        exit(1);
    }
    while(getline(infile,temp))
    {

        string first,second;
        unsigned index=0;
        for(unsigned i=0; i<temp.size(); ++i)
        {
            if((temp[i]==' '&&temp[i+1]==' ')||(temp[i]=='\t'&&temp[i+1]==' ')||(temp[i]==' '&&temp[i+1]=='\t') ||(temp[i]=='\t'&&temp[i+1]=='\t'))continue;
            else if(temp[i]!=':'&&temp[i]!='\t')
                first+=temp[i];
            else if(temp[i]=='\t')
                first+=' ';
            else if(temp[i]==':')
            {
                index=i;
                break;
            }
        }


            while(1)
            {
                string::iterator p=first.begin();
                if(*p==' ')
                first.erase(p);
                else break;
            }


            for(unsigned i=index+1; i<temp.size(); ++i)
            {

                if((temp[i]==' '&&temp[i+1]==' ')||temp[i]=='\t')continue;
                else second+=temp[i];
            }

            while(1)
            {string::iterator q=second.begin();
            if(*q==' ')
                second.erase(q);
            else break;
            }
            outfile<<setw(n-1)<<left<<first<<": "<<left<<second<<endl;

        }

    infile.close();
    outfile.close();
    return 0;
}

我在写这道题的时候,清除空格就花了很长时间,尤其是没想明白为什么处理之后还会保留两个连续的空格,最后才想到有四种情况。另外,不用迭代器和erase函数,清理开头空格会麻烦一些。

  • 3
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值