输入两个字符串,从第一个字符串中删除第二个字符串中的所有字符
比如输入“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;
}