【问题描述】
英文电影中参演人员名单一般以某种方式进行排版显示。给定一个未排版的文件listin.txt,该文件中每行参演人员名单由冒号ldquo:rdquo分隔成前后两部分,但格式杂乱无章,单词(由除空格和水平制表符之外的其它字符构成)之间可能有多个空格或水平制表符分隔。编写程序,要求将其按如下排版规则排版输出到另一个文件listout.txt中:
1.从标准输入整数,作为排版后所有各行冒号在一行中的固定位置,输入的整数肯定大于排版后所有各行冒号前的字符个数,位置从1开始计数;
2.冒号左边的单词串以行头为基准左对齐,左边的最后一个单词与冒号之间以空格填充;冒号右边的单词串以冒号基准左对齐,最后一个单词后只有回车换行符,不再有其它字符;
3.冒号左右两边的单词间都只有一个空格分隔,并且要求冒号两边至少各有一个空格。
假设输入文件中每行字符个数不超过100。
【输入形式】
待排版的参演人员名单从当前目录下的listin.txt文件中读入,表示冒号位置的整数从标准输入读入。
【输出形式】
排版后的参演人员名单输出到当前目录下的listout.txt中。
【输入样例】
假设文件listin.txt内容为:
表示冒号固定位置的整数为:
40
【输出样例】
文件listout.txt中的内容应为:
【样例说明】
输入的文件listin.txt中有五行参演人员名单,要求排版后冒号位于第40个字符的位置,按照上述排版规则输出到文件listout.txt中。
#include <fstream>
#include <iostream>
#include <string>
using namespace std;
int main() {
ifstream fin("listin.txt");
ofstream fout("listout.txt");
string a, b, c;
int loc = 0, i = 0;
cin >> loc;
while (!fin.eof()) {
getline(fin, a);
if (a == "") {
continue;
}
for (i = 0;i < a.length();i++) {//先把所有制表符换成空格一起处理
if (a[i] == '\t') {
a.replace(i, 1, " ");
}
}
for (i = 0;;i++) { //处理开头的空格
if (a[i] == ' ' ){
a.replace(i, 1, "");
i--;
}
else {
break;
}
}
for (;;i++) { //删除中间空格
if (a[i] == ' ' && a[i + 1] == ' ') {
a.replace(i, 1, "");
i--;
}
if (a[i] == ':') {
break; //找到冒号为止
}
}
b = a.substr(0, i); //冒号前的字符串
c = a.substr(i + 1); //冒号后的字符串
for (i = 0;;i++) { //处理冒号后的空格
if (c[i] == ' ') {
c.replace(i, 1, "");
i--;
}
else {
break;
}
}
for (;i < c.length();i++) {
if (c[i] == ' ' && c[i + 1] == ' ') {//这里调试了好久,只因为把c打成a了,吐血
c.replace(i, 1, "");
i--;
}
}
for (i = c.length() - 1;i > 0;i -- ) {//删除后面的空格
if (c[i] == ' ') { //这里又调试了好久,又是因为把c打成a了,emmmmm
c.replace(i, 1, "");
}
else {
break;
}
}
fout.width(loc-1);
fout.setf(ios::left);
fout.fill(' ');
fout << b << ": " << c << endl;
}
fin.close();
fout.close();
}
总结:
这道题相当不好用字符串行处理,要考虑的地方非常多,包括换行符的识别,结尾删除空格,最简单的方法应该是依次读取单词,再按照格式输出。
但目前还不知道除了getline有什么识别换行的方法。以后有时间再试试另一种方式。
注意:
1.复制粘贴代码要改全,a和c就别错了
2.缺点是处理复杂,太多循环,时间上看很糟糕