【问题描述】
英文电影中参演人员名单一般以某种方式进行排版显示。给定一个未排版的文件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函数,清理开头空格会麻烦一些。