// 编写程序entab,将空格串替换为最少数量的制表符和空格,但要保持单词之间的间隔不变
#include<stdio.h>
#define stopbit 8 /* 制表符终止位 */
#define MAXLINE 1001 /* 允许的输入行的最大长度 */
typedef signed char int8_t;
static void entab(int8_t s[]);
/* 空格串替换为最少数量制表符,那么空格串必须包含一个或n个制表符终止位 */
int main(void)
{
int8_t line[MAXLINE] = { 0 };//当前的输入行
int c, i = 0;
freopen("C:\\\\Users\\wwwzh\\Desktop\\data.in", "r", stdin);
freopen("C:\\\\Users\\wwwzh\\Desktop\\data.out", "w", stdout);
while ((c = getchar()) != EOF)
{
line[i] = c;
if (c == '\n')
{
i++;
line[i] = '\0';
entab(line);
printf("%s", line);
i = -1;
}
i++;
}
return 0;
}
extern int detab(char s[]);
/* 如果空格串中包含制表符终止位,替换成等价的制表符和空格 */
static void entab(int8_t str[])
{
int8_t s[MAXLINE] = { 0 };
int i = 0, j = 0;
//1.复制一份并把字符串全部展开
while (str[i] != '\0')
{
s[i] = str[i];
i++;
}
detab(s);//需要把整个输入展开
//2.改变字符串
i = 0;
while (s[i] != '\0')
{
int fstr = i, estr = fstr;//空格串起始坐标和结束坐标
int num = 0;//空格串转换需要的空格数
int count = 0;//空格串转换需要的制表符数
if (s[fstr] == ' ')
{
int estop = fstr + 1;//最后一个制表符终止位下标位置
while (s[estr] == ' ')
{
int k = stopbit - estr % stopbit - 1;//字符位置7,15...为终止位
if (k == 0)
{
estop = estr;
count++;
}
estr++;
}
estr--;//最终空格串结束坐标
num = estr - estop;
if (count > 0)//空格串存在转换
{
int k = fstr;
while (count > 0)
{
s[k] = '\t';
count--;
k++;
}
while (num > 0)
{
s[k] = ' ';
num--;
k++;
}
while (k <= estr)
{
s[k] = -1;
k++;
}
}
i = estr;
}
i++;
}
//3.把改变的字符串赋给原字符串
i = 0;
while (s[i] != '\0')
{
if (s[i] != -1)
{
str[j] = s[i];
j++;
}
i++;
}
str[j] = '\0';
}
2个说明
第1个是为什么要把制表符先转换成空格再换成制表符,如果直接处理空格串(前面有\t)
那应该也能处理但应该麻烦
第2个:因为用的是signed char,所以对中文(虽然显示的是一个字符,但实际处理它要占2个字符)的处理估计有问题
但把它转换成等价字符的英文就没事