题目:替换空格,请实现一个函数,把字符串中的每个空格替换成"%20"。例如输入“we are happy”,则输成“we%20are%20happy”。
看到这个题目,我们首先需要发现,一个空格是一个字符,而%20是三个字符,所以字符串会变长。如果直接去进行修改,很容易就会导致空格后面的原有字符串被覆盖掉。
现在我们来考虑一下怎么做替换操作,最直观的做法是遍历一遍,没遇到一次空格就做一次替换,而且我们需要将空格后面的字符串向后移动两格,否则会导致覆盖的。但是,这种方法显然是比较复杂的,每一次移动需要移动n个字符,n个字符移动,所以时间复杂度为O(N^2),因此这种方法并不建议。
现在我们来考虑另外一种方法,我们先遍历一遍字符串,统计出空格的个数,然后由此计算出需要的总的字符串的长度。然后我们倒着从字符串的后面开始复制和替换。一个走原字符串,一个走加了长度的字符串,碰到一个空格,就开始处理,将新字符串的连续的几个位置插上需要被替换的字符,如果没有遇到空格就直接将原字符串的值赋给新的字符串。
void Replace(char* str, int len)
{
int i = 0;
int j = 0;
int count = 0;
while (i < len)//遇到空格就计数
{
if (str[i] == ' ')
{
count++;
}
++i;
}
i = len;
j = 2 * count + len;//修改字符串的长度,每一次的空格都加上2个长度
while (i != j && i >= 0)//i!=j这个条件是为了减少最后一次空格前的字符替换,进行一些优化
{
if (str[i] == ' ')//倒着插入可以减少移动的次数,反正原本也是需要往前走的
{
str[j--] = '0';
str[j--] = '2';
str[j--] = '%';
i--;
}
else
{
str[j] = str[i];
j--;
i--;
}
}
}
另外,我这里给出一种取巧的方式,直接将空格打印成字符串
void Replace(char* str)
{
char ch[] = "%20";
while (*str != '\0')
{
if (*str == ' ')
{
printf("%s", ch);
*str++;
}
printf("%c", *str);
*str++;
}
}
当然,这里这个取巧的方式有一个前提条件,你字符串的空间必须开的足够大,不能因为空间不足而溢出。