Bootstrap

输入两个字符串,从第一个字符串中删除第二个字符串中的所有字符(详细)

输入两个字符串,从第一个字符串中删除第二个字符串中的所有字符
比如输入“They are students."
删除之后的第一个字符串变成"The r stdnts."

思路是利用每个字符都有其对应的ASCII码值,将需要删除的字符的ascii为下标储存在数组当中。
利用两个指针分别指向原字符串和需要删除的字符串。
因为我们已经将需要删除的字符串中的字符所对应的ASCII码作为下标储存在数组当中,所以在比较的时候只需要看原字符串中的字符所对应的ASCII码作为下标在数组中是否为1.
这样说可能不好理解。看一下具体例子

假如原字符串为

char str1[] = "They aaaare students.";

需要删除的为

char str2[] = "aeiou";

然后我们需要一个数组储存结果(将其初始化为0)

int a[256] = { 0 };

两个指针分别指向这两个字符串

char* pFast = str1;
char* pSlow = str2;

我们想实现的是在字符串str1中找到‘a’,‘e’,‘i’,‘o’,‘u’这几个字符并删除
第一步我们想在str1中找到所有的字符‘a’并删除
也就是说我们所有想删除的字符都在str2中

接下来我们把需要删除的字符对应的ASCII码作为下标,然后储存在数组中
利用一个for循环

for (i = 0; i < n; ++i){
	a[str2[i]] = 1;
}

str2[i]对应的是所有需要在str1中删除的字符
将其对应的ASCII码作为下标也就是 a[str2[i]];
将其赋为1,表示该字符需要删除

接下来实现代码的核心部分

while (*pFast){
	if (a[*pFast] == 1){
		*pFast++;
	}
	*pSlow++ = *pFast++;
}

因为我们需要删除的是str1中的字符,所以这里的 while(pFast)是为了遍历str1
接下来 if (a[pFast] == 1) 说明我们要删除这个字符。我们给pFast++,让pFast指针跳过这个字符,指到下一个字符,然后再赋给pSlow。
这样做本质上是遍历str1,如果这个字符不需要删除,就拿过来赋给pSlow,然后pFast和pSlow同时自增
如果需要删除,先给pFast自增,再
pSlow++ = *pFast++;(自增同时赋值)

别忘了最后

*pSlow = '\0';

给字符串结尾!

但是这样的思路还存在一些问题。先看上面这个思路的完整代码,我们再来修改

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

void DeleteStr(char* str1, char* str2){
	char* pFast = str1;
	char* pSlow = str2;
	int a[256] = { 0 };
	int i;
	int n = strlen(str2);
	for (i = 0; i < n; ++i){
		a[str2[i]] = 1;
	}
	while (*pFast){
		if (a[*pFast] == 1){
			*pFast++;
		}
		*pSlow++ = *pFast++;
	}

	*pSlow = '\0';
}
int main(){
	char str1[] = "They are students.";
	char str2[] = "aeiou";
	DeleteStr(str1, str2);
	printf("%s\n", str2);
	system("pause");
	return 0;
}

这样去做题目中的要求我们可以实现,但是拓展之后,如果相同的字符连续出现,我们无法做到连续删除

比如原字符串变成这样

char str1[] = "They aaaare students.";

仔细想想,来看我们上面的核心代码

while (*pFast){
	if (a[*pFast] == 1){
		*pFast++;
	}
	*pSlow++ = *pFast++;
}

我们在确认一个字符需要删除之后,直接就跳过该字符,下一个字符没有进行判断是否应该删除就进行了赋值
所以要将这段代码进行修改

while (*pFast){
	if (a[*pFast] == 0){
		*pSlow = *pFast;
		pSlow++;
	}
	pFast++;
}

这次我们判断

if (a[*pFast] == 0){}

就说明不需要删除可以赋值
直接

*pSlow = *pFast

紧接着下来的一步我们不着急赋值,因为我们最终使用pSlow保存的结果,所以每赋值一次就先给pSlow++,这是pSlow++放在if语句中的原因。试想如果将pSlow++放在if语句之外,那么我们没有给pSlow赋值,指针也会不断往前走,这样肯定会出问题。
先给pSlow++之后,在if语句之外给先给pFast++
这时开始第二次循环
这样做就能够实现在每一次赋值之前进行判断,就算相同字符连续出现也可以实现删除
完整代码如下

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
void DeleteStr(char* str1, char* str2){
	char* pFast = str1;
	char* pSlow = str2;
	int a[256] = { 0 };
	int i;
	int n = strlen(str2);
	for (i = 0; i < n; ++i){
		a[str2[i]] = 1;
	}
	while (*pFast){
		if (a[*pFast] == 0){
			*pSlow = *pFast;
			pSlow++;
		}
		pFast++;
	}
	*pSlow = '\0';
}
int main(){
	char str1[] = "They aaaare students.";
	char str2[] = "aeiou";
	DeleteStr(str1, str2);
	printf("%s\n", str2);
	system("pause");
	return 0;
}
;