去除字符串中重复字符
来源:互联网 发布:linux cp强制复制 编辑:程序博客网 时间:2024/06/05 06:41
题目http://www.cricode.com/260.html
设计算法并写出代码移除字符串中重复的字符,不能使用额外的缓存空间。注意: 可以使用额外的一个或两个变量,但不允许额外再开一个数组拷贝。
进一步地,
为你的程序写测试用例。
解答
这道题目其实是要你就地(in place)将字符串中重复字符移除。你可以向面试官问清楚, 不能使用额外的一份数组拷贝是指根本就不允许开一个数组,还是说可以开一个固定大小, 与问题规模(即字符串长度)无关的数组。
如果根本就不允许你再开一个数组,只能用额外的一到两个变量。那么,你可以依次访问 这个数组的每个元素,每访问一个,就将该元素到字符串结尾的元素中相同的元素去掉( 比如置为’’).时间复杂度为O(n2 ),代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
void removeDuplicate(char s[])
{
int len = strlen(s);
if(len < 2) return;
int p = 0;
for(int i=0; i < len; ++i)
{
if(s[i] != '')
{
s[p++] = s[i];
for(int j=i+1; j < len; ++j)
if(s[j]==s[i])
s[j] = '';
}
}
s[p] = '';
}
如果可以开一个固定大小,与问题规模(即字符串长度)无关的数组,那么可以用一个数组来 表征每个字符的出现(假设是ASCII字符,则数组大小为256),这样的话只需要遍历一遍字符 串即可,时间复杂度O(n)。代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
voidremoveDuplicate(chars[])
{
intlen=strlen(s);
if(len<2)return;
boolc[256];
memset(c,0,sizeof(c));
intp=0;
for(inti=0;i<len;++i)
{
if(!c[s[i]])
{
s[p++]=s[i];
c[s[i]]=true;
}
}
s[p]='';
}
如果字符集更小一些,比如只是a-z,即字符串里只包含小写字母,那么使用一个int变量中 的每一位来表征每个字符的出现,一样可以在O(n)的时间里移除重复字符,而且还不需要额 外开一个数组。代码如下:
void removeDuplicate(char s[])
{
int len = strlen(s);
if(len < 2) return;
int check = 0, p = 0;
for(int i=0; i < len; ++i)
{
int v = (int)(s[i]-'a');
if((check & (1 << v))==0)
{
s[p++] = s[i];
check |= (1 << v);
}
}
s[p] = '';
}
测试用例:
- 不包含重复字符的字符串,比如:abcd
- 字符串全是重复字符,比如:aaaa
- 空字符串
- 重复字符连续出现,比如:aaabbb
- 重复字符不连续出现,比如:abababa
完整代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
#include <iostream>
#include <cstring>
usingnamespacestd;
stringremoveDuplicate1(strings)
{
intcheck=0;
intlen=s.length();
if(len<2)returns;
stringstr="";
for(inti=0;i<len;++i)
{
intv=(int)(s[i]-'a');
if((check&(1<<v))==0)
{
str+=s[i];
check|=(1<<v);
}
}
returnstr;
}
stringremoveDuplicate2(strings)
{
intlen=s.length();
if(len<2)returns;
stringstr="";
for(inti=0;i<len;++i)
{
if(s[i]!='')
{
str+=s[i];
for(intj=i+1;j<len;++j)
if(s[j]==s[i])
s[j]='';
}
}
returnstr;
}
voidremoveDuplicate3(chars[])
{
intlen=strlen(s);
if(len<2)return;
intp=0;
for(inti=0;i<len;++i)
{
if(s[i]!='')
{
s[p++]=s[i];
for(intj=i+1;j<len;++j)
if(s[j]==s[i])
s[j]='';
}
}
s[p]='';
}
voidremoveDuplicate4(chars[])
{
intlen=strlen(s);
if(len<2)return;
boolc[256];
memset(c,0,sizeof(c));
intp=0;
for(inti=0;i<len;++i)
{
if(!c[s[i]])
{
s[p++]=s[i];
c[s[i]]=true;
}
}
s[p]='';
}
voidremoveDuplicate5(chars[])
{
intlen=strlen(s);
if(len<2)return;
intcheck=0,p=0;
for(inti=0;i<len;++i)
{
intv=(int)(s[i]-'a');
if((check&(1<<v))==0)
{
s[p++]=s[i];
check|=(1<<v);
}
}
s[p]='';
}
intmain()
{
strings1="abcde";
strings2="aaabbb";
strings3="";
strings4="abababc";
strings5="ccccc";
cout<<removeDuplicate1(s1)<<" "<<removeDuplicate2(s1)<<endl;
cout<<removeDuplicate1(s2)<<" "<<removeDuplicate2(s2)<<endl;
cout<<removeDuplicate1(s3)<<" "<<removeDuplicate2(s3)<<endl;
cout<<removeDuplicate1(s4)<<" "<<removeDuplicate2(s4)<<endl;
cout<<removeDuplicate1(s5)<<" "<<removeDuplicate2(s5)<<endl;
charss1[]="abcde";
charss2[]="aaabbb";
charss3[]="";
charss4[]="abababc";
charss5[]="ccccc";
removeDuplicate5(ss1);
removeDuplicate5(ss2);
removeDuplicate5(ss3);
removeDuplicate5(ss4);
removeDuplicate5(ss5);
cout<<ss1<<" "<<ss2<<" "<<ss3<<" "<<ss4<<" "<<ss5<<endl;
return0;
}
【JAVA参考代码】
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
publicstaticvoidremoveDuplicates(char[]str){
if(str==null)return;
intlen=str.length;
if(len<2)return;
inttail=1;
for(inti=1;i<len;++i){
intj;
for(j=0;j<tail;++j){
if(str[i]==str[j])break;
}
if(j==tail){
str[tail]=str[i];
++tail;
}
}
str[tail]=0;
}
0 0