摘要
剑指offer面试题 4:
实现一个函数,此函数可以将字符串中的空格替换,,,,,
例如字符串"we are happy.",,,,替换之后得到的字符串是"we%20are%20happy."
看到这个题我们第一种做法就是
找到一个空格之后吗,,,将他替换成是 %20;
但要将他替换成是%20的话;;;;就必须要将后面的字符先向后面移动;;;;
实现代码
//方法一 :死方法 遇到一个空格就把空格后面的字符向后移,,,知道结束;
//str表示的字符串,,,n表示的就是数组 长度
//假设数组的长度足够长
void ReplaceBlank(char * src,const size_t n)
{
assert(src);//先断言 。预防是个空串
int length = strlen(src);//表示的是字符串的实际长度 算上/0
char *str = src;
while(*str)
{
if(*str == ' ')
{
//遇到一个空格之后 ,,,先将空格之后的字符向后移两位
assert(n >=(length+3));//如果要是空间不够用的话
char* tmp = src + length;
while(str != tmp)
{
*(tmp +2) = *tmp;
tmp--;
}
*str ='%';
*(++str) = '2';
*(++str) ='0';
length = strlen(src);
}
++str;
}
}
这种方法看上去好理解,,,但是此种算法的时间复杂度相当于O(N*N)
如果要是面试管问的是这个问题的话,,,那么他是不会满意的。。。。
所以肯定还有优化的空间》》》》
上述方法就是因为每次遇到空格都要移动,,,,,因此我们可以只需要移一次
只要知道当前空格的个数就知道要移动的位数
我们可以先遍历一般的到空格的个数,,在移动
代码
//方法2:我们可以先遍历一遍字符串,判断总共有几个空格,,就可以直接的得到移动的位数
//此方法的时间复杂度为 O(1),,因为只遍历了一遍并且移动了一次
void ReplaceBlankS(char * src,const size_t n)
{
assert(src);//同样的先断言,,,判断字符串势是不是空串
char * str = src;
int count = 0;//表示的是空格的个数
while(*str)
{
if(*(str++)==' ')
{
count++;
}
}
assert((strlen(src)+count*2)+1 <= n)//断言数组要足够长
//此时的str表示的是/0,,,也就是最后一个字符
//从后往前来判断,
while(str !=src)
{
if(*str != ' ')
{
*(str + 2*count) = *str;
}
else
{
*(str + 2*count) ='0';
*(str + 2*count - 1) = '2';
*(str + 2*count - 2) = '%';
count--;//遇到一个空格之后 ,,那么前面的空格就会减少 ,,,移动的位数就会减少
}
str--;
}
}