您的位置:首页 > 其它

字符串逆序的算法汇总

2013-09-24 20:52 169 查看
很早就准备写一个字符串系列的面试题,本来已经写好了,大概有十几道题,但是写完才发现,文章好长,连我自己都没有耐心读下去了,索性就将其拆分成几个系列,一来分开后篇幅变小,看起来比较方便。二来也更有针对性,便于精雕细作。比如这篇,在原来的文章中只占很小的篇幅,但是独立出来才发现,东西也不少。既然是第一篇,就来个最最简单的字符串逆序吧。

字符串逆序可以说是最经常考的题目。这是一道入门级的题目,相信80%的程序员经历过这道题。给定一个字符串s,将s中的字符顺序颠倒过来,比如s="abcd",逆序后变成s="dcba"。


普通逆序

基本上没有这么考的,放在这里主要是为了和后面的原地逆序做个对比。很简单,直接分配一个与原字符串等长的字符数组,然后反向拷贝一下即可。

01
char
*Reverse(
char
*s)
02
{
03
//将q指向字符串最后一个字符
04
char
*q
= s ;
05
while
(*q++)
06
;
07
q
-= 2 ;
08
09
//分配空间,存储逆序后的字符串。
10
char
*p
=
new
char
[
sizeof
(
char
)
* (q - s +2)] ;
11
char
*r
= p ;
12
13
//
逆序存储
14
while
(q
>= s)
15
*p++
= *q-- ;
16
*p
=
'\0'
;
17
18
return
r
;
19
}


原地逆序

英文叫做in-place reverse。这是最常考的,原地逆序意味着不允额外分配空间,主要有以下几种方法,思想都差不多,就是将字符串两边的字符逐个交换,如下图。给定字符串"abcdef",逆序的过程分别是交换字符a和f,交换字符b和e,交换字符c和d。

设置两个指针,分别指向字符串的头部和尾部,然后交换两个指针所指的字符,并向中间移动指针直到交叉。

01
char
*Reverse(
char
*s)
02
{
03
//
p指向字符串头部
04
char
*p
=s ;
05
06
//
q指向字符串尾部
07
char
*q
= s ;
08
while
(*q)
09
++q
;
10
q
-- ;
11
12
//
交换并移动指针,直到p和q交叉
13
while
(q
>p)
14
{
15
char
t
= *p ;
16
*p++
= *q ;
17
*q--
= t ;
18
}
19
20
return
s
;
21
}
用递归的方式,需要给定逆序的区间,调用方法:Reverse(s, 0, strlen(s)) ;

01
//
对字符串s在区间left和right之间进行逆序,递归法
02
char
*Reverse(
char
*s,
int
left,
int
right
)
03
{
04
if
(left
>= right)
05
return
s
;
06
07
char
t
= s[left] ;
08
s[left]
= s[right] ;
09
s[right]
= t ;
10
11
Reverse(s,
left +1, right - 1) ;
12
}
非递归法,同样指定逆序区间,和方法一没有本质区别,一个使用指针,一个使用下标。

01
//
对字符串str在区间left和right之间进行逆序
02
char
*Reverse(
char
*s,
int
left,
int
right
)
03
{
04
while
(
left <right )
05
{
06
char
t
= s[left] ;
07
s[left++]
= s[right] ;
08
s[right--]
= t ;
09
}
10
11
return
s
;
12
}


不允许临时变量的原地逆序

上面的原地逆序虽然没有额外分配空间,但还是使用了临时变量,严格的说也算是额外的空间吧,如果再严格一点,连临时变量也不允许的话,主要有下面两种方法。一是异或操作,因为异或操作可以交换两个变量而无需借助第三个变量,二是使用字符串的结束符'\0'所在的位置作为交换空间,这样有个局限,就是只适合以'\0'结尾的字符串,对于不支持这种字符串格式的语言,就不能使用了。

使用字符串结束符'\0'所在的位置作为交换空间:

01
//
使用字符串结束符'\0'所在的位置作为交换空间
02
char
*
Reverse(
char
*
s)
03
{
04
char
*
r = s ;
05
06
//
令p指向结束符
07
char
*
p = s;
08
while
(*p
!=
'\0'
)
09
++p
;
10
11
//
令q指向字符串最后一个字符
12
char
*
q = p - 1;
13
14
//
使用p作为交换空间逐个交换字符
15
while
(q
>s)
16
{
17
*p
=*q ;
18
*q--
= *s ;
19
*s++
= *p ;
20
}
21
22
*p
=
'\0'
;
//
恢复结束符
23
24
return
r
;
25
}
使用异或操作

01
//
使用异或操作对字符串s进行逆序
02
char
*
Reverse(
char
*
s)
03
{
04
char
*
r = s ;
05
06
//令p指向字符串最后一个字符
07
char
*
p = s;
08
while
(*(p
+1) !=
'\0'
)
09
++p
;
10
11
//
使用异或操作进行交换
12
while
(p
>s)
13
{
14
*p
=*p ^ *s ;
15
*s
= *p ^ *s ;
16
*p
=*p-- ^ *s++;
17
}
18
19
return
r
;
20
}


按单词逆序

给定一个字符串,按单词将该字符串逆序,比如给定"This is a sentence",则输出是"sentence a is This",为了简化问题,字符串中不包含标点符号。 分两步:

先按单词逆序得到"sihT si a ecnetnes"
再整个句子逆序得到"sentence a is This"

对于步骤一,关键是如何确定单词,这里以空格为单词的分界。当找到一个单词后,就可以使用上面讲过的方法将这个单词进行逆序,当所有的单词都逆序以后,将整个句子看做一个整体(即一个大的包含空格的单词)再逆序一次即可,如下图所示,第一行是原始字符换,第二行是按单词逆序后的字符串,最后一行是按整个句子逆序后的字符串。

01
//
对指针p和q之间的所有字符逆序
02
void
ReverseWord(
char
*
p,
char
*
q)
03
{
04
while
(p
<q)
05
{
06
char
t
= *p ;
07
*p++
= *q ;
08
*q--
= t ;
09
}
10
}
11
12
//
将句子按单词逆序
13
char
*
ReverseSentence(
char
*s)
14
{
15
//
这两个指针用来确定一个单词的首尾边界
16
char
*p
=s ;
//
指向单词的首字符
17
char
*q
= s ;
//
指向空格或者 '\0'
18
19
while
(*q
!=
'\0'
)
20
{
21
if
(*q
==
'
'
)
22
{
23
ReverseWord(p,
q - 1) ;
24
q++
;
//
指向下一个单词首字符
25
p
= q ;
26
}
27
else
28
q++
;
29
}
30
31
ReverseWord(p,
q - 1) ;
//
对最后一个单词逆序
32
ReverseWord(s,
q - 1) ;
//
对整个句子逆序
33
34
return
s
;
35
}


逆序打印

还有一类题目是要求逆序输出,而不要求真正的逆序存储。这题很简单,有下面几种方法,有的方法效率不高,这里仅是提供一个思路而已。先求出字符串长度,然后反向遍历即可。

1
void
ReversePrint(
const
char
*
s)
2
{
3
int
len
=
strlen
(s)
;
4
for
(
int
i
= len - 1;i >= 0;--i)
5
cout
<<s[i];
6
}
如果不想求字符串的长度,可以先遍历到末尾,然后在遍历回来,这要借助字符串的结束符'\0'。

01
void
ReversePrint(
const
char
*
s)
02
{
03
const
char
*
p = s ;
04
05
while
(*p)
06
*p++
;
07
08
--p
;
//while结束时,p指向'\0',这里让p指向最后一个字符
09
10
while
(p
>= s)
11
{
12
cout
<<*p ;
13
--p
;
14
}
15
}
对于上面第二种方法,也可以使用递归的方式完成。

1
void
ReversePrint(
const
char
*
s)
2
{
3
if
(*(s
+1) !=
'\0'
)
4
ReversePrint(s
+1) ;
5
cout
<<*s ;
6
}
== THE END==

Happy coding!

原文转自:http://www.nowamagic.net/algorithm/algorithm_StringReverse.php
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: