- 浏览: 21835 次
- 性别:
- 来自: 南京
最新评论
很早就准备写一个字符串系列的面试题,本来已经写好了,大概有十几道题,但是写完才发现,文章好长,连我自己都没有耐心读下去了,索性就将其拆分成几个系列,一来分开后篇幅变小,看起来比较方便。二来也更有针对性,便于精雕细作。比如这篇,在原来的文章中只占很小的篇幅,但是独立出来才发现,东西也不少。既然是第一篇,就来个最最简单的字符串逆序吧。
字符串逆序可以说是最经常考的题目。这是一道入门级的题目,相信80%的程序员经历过这道题。给定一个字符串s,将s中的字符顺序颠倒过来,比如s="abcd",逆序后变成s="dcba"。
普通逆序
基本上没有这么考的,放在这里主要是为了和后面的原地逆序做个对比。很简单,直接分配一个与原字符串等长的字符数组,然后反向拷贝一下即可。
view sourceprint?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。
设置两个指针,分别指向字符串的头部和尾部,然后交换两个指针所指的字符,并向中间移动指针直到交叉。
view sourceprint?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)) ;
view sourceprint?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 }
非递归法,同样指定逆序区间,和方法一没有本质区别,一个使用指针,一个使用下标。
view sourceprint?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'所在的位置作为交换空间:
view sourceprint?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 }
使用异或操作
view sourceprint?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"
对于步骤一,关键是如何确定单词,这里以空格为单词的分界。当找到一个单词后,就可以使用上面讲过的方法将这个单词进行逆序,当所有的单词都逆序以后,将整个句子看做一个整体(即一个大的包含空格的单词)再逆序一次即可,如下图所示,第一行是原始字符换,第二行是按单词逆序后的字符串,最后一行是按整个句子逆序后的字符串。
view sourceprint?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 }
逆序打印
还有一类题目是要求逆序输出,而不要求真正的逆序存储。这题很简单,有下面几种方法,有的方法效率不高,这里仅是提供一个思路而已。先求出字符串长度,然后反向遍历即可。
view sourceprint?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'。
view sourceprint?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 }
对于上面第二种方法,也可以使用递归的方式完成。
view sourceprint?1 void ReversePrint(const char* s)
2 {
3 if(*(s + 1) != '\0')
4 ReversePrint(s + 1) ;
5 cout << *s ;
6 }
== THE END==
Happy coding!
字符串逆序可以说是最经常考的题目。这是一道入门级的题目,相信80%的程序员经历过这道题。给定一个字符串s,将s中的字符顺序颠倒过来,比如s="abcd",逆序后变成s="dcba"。
普通逆序
基本上没有这么考的,放在这里主要是为了和后面的原地逆序做个对比。很简单,直接分配一个与原字符串等长的字符数组,然后反向拷贝一下即可。
view sourceprint?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。
设置两个指针,分别指向字符串的头部和尾部,然后交换两个指针所指的字符,并向中间移动指针直到交叉。
view sourceprint?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)) ;
view sourceprint?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 }
非递归法,同样指定逆序区间,和方法一没有本质区别,一个使用指针,一个使用下标。
view sourceprint?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'所在的位置作为交换空间:
view sourceprint?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 }
使用异或操作
view sourceprint?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"
对于步骤一,关键是如何确定单词,这里以空格为单词的分界。当找到一个单词后,就可以使用上面讲过的方法将这个单词进行逆序,当所有的单词都逆序以后,将整个句子看做一个整体(即一个大的包含空格的单词)再逆序一次即可,如下图所示,第一行是原始字符换,第二行是按单词逆序后的字符串,最后一行是按整个句子逆序后的字符串。
view sourceprint?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 }
逆序打印
还有一类题目是要求逆序输出,而不要求真正的逆序存储。这题很简单,有下面几种方法,有的方法效率不高,这里仅是提供一个思路而已。先求出字符串长度,然后反向遍历即可。
view sourceprint?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'。
view sourceprint?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 }
对于上面第二种方法,也可以使用递归的方式完成。
view sourceprint?1 void ReversePrint(const char* s)
2 {
3 if(*(s + 1) != '\0')
4 ReversePrint(s + 1) ;
5 cout << *s ;
6 }
== THE END==
Happy coding!
发表评论
-
KMP快速字符串查找算法
2011-08-25 19:29 668在C/C++语言编程过程中,一般的字符串搜索操作都是通过标准库 ... -
求解最大公约数问题
2011-08-25 19:27 693最大公因数,又称最大公约数。是指 [n(≧2)个自然数 a1, ... -
堆排序(Heap Sort)算法学习
2011-08-25 19:26 1084在程序设计相关领域, ... -
整数拆分问题的动态规划解法
2011-08-25 19:26 3070输入n,和k,问将n用1到k这k个数字进行拆分,有多少种拆分方 ... -
背包问题介绍与分析
2011-08-25 19:24 1030背包问题是在1978年由Merkel和Hellman提出的。它 ... -
求平方根sqrt()函数的底层算法效率问题
2011-08-25 19:23 1291我们平时经常会有一些数据运算的操作,需要调用sqrt,exp, ... -
面试中常见的一些算法问题
2011-08-25 19:22 697Problem 1 : Is it a loop ? ( ... -
各种排序算法的C++实现与性能比较
2011-08-25 19:21 920排序是计算机算法中非常重要的一项,而排序算法又有不少实现方法, ... -
背包问题之硬币找零问题
2011-08-25 19:19 1173设有6 种不同面值的硬 ... -
求能被1到20的数整除的最小正整数
2011-08-25 19:18 1369求能被1到20的数整除的最小正整数。最直觉的方法是求1到20这 ... -
买书折扣最优惠问题解法
2011-08-25 19:17 752题目:在节假日的时候 ... -
二叉树中的最近公共祖先问题
2011-08-25 19:16 1323题目:要求寻找二叉树中两个节点的最近的公共祖先,并将其返回。 ... -
判断一个整数是否是2的N次方
2011-08-25 19:04 1822题目:给定一个整数num,判断这个整数是否是2的N次方。比如, ... -
计算从1到N中1的出现次数
2011-08-25 18:59 598给定一个十进制整数N, ... -
KMP快速字符串查找算法
2011-08-25 18:57 965在C/C++语言编程过程中,一般的字符串搜索操作都是通过标准库 ... -
快速排序的递归实现
2011-08-25 18:54 756快速排序是对冒泡排序的一种改进。它的基本思想是:通过一次排序将 ... -
数字1亿里面有多少个1呢
2011-08-25 18:52 739乍看这题真够唬人的,群里看到这个题目后争先恐后的说看法。最简单 ... -
最大子序列、最长公共子串、最长公共子序列
2011-08-25 18:33 791最大子序列 最大子序列是要找出由数组成的一维数组中和最大的连续 ... -
一道关于男女比例的面试题
2011-08-25 16:56 1037阿里巴巴的一道面试题:说澳大利亚的父母喜欢女孩,如果生出来的第 ...
相关推荐
本文将深入探讨如何使用Python实现字符串逆序输出,以及相关的字符串遍历、翻转和排序操作。 首先,让我们看看如何简单地实现字符串的逆序输出。在Python中,可以通过索引和切片操作轻松完成这一任务。例如,如果...
这是一种简单的字符串处理算法,可以扩展到其他数字格式的处理。 汉诺塔问题(程序1-6)是一个经典的递归问题,目标是将所有盘子从一个柱子移动到另一个柱子,每次只能移动一个盘子,并且大盘子不能位于小盘子之上...
- **KMP算法**:处理字符串匹配问题,避免不必要的回溯。 - **Manacher's Algorithm**:在线性时间内找出字符串中的最长回文子串。 8. **数据结构**: - **栈**:后进先出(LIFO)的数据结构,常用于表达式求值...
3. 字符串逆序输出:考察字符串处理,可能需要用到字符串指针和反向遍历字符串的方法。 4. 数组中最大值和最小值的交换:这涉及到数组操作和比较操作,需要找到数组中的最大值和最小值并进行交换。 5. 字符串转...
VB提供了丰富的字符串处理函数,如Mid、Left、Right、InStr等,可以进行字符串截取、查找、替换等操作。 六、文件操作 VB支持对文本文件和二进制文件的读写,如Open、Close、Input#、Write#、Get#、Put#等语句。 ...
文章还提到了一些与本主题相关的其他Python学习资源,比如《Python函数使用技巧总结》、《Python数据结构与算法教程》、《Python字符串操作技巧汇总》、《Python入门与进阶经典教程》及《Python文件与目录操作技巧...
6. **逆序输出字符串**:反转字符串是常见的字符串操作,可以使用StringBuilder类的Reverse方法,或者创建新字符串并从后向前遍历原字符串添加字符。 7. **取两个数组的相同元素**:比较两个数组并找到相同的元素,...
strcpy函数用于复制一个字符串到另一个字符串,注意目标字符串需要有足够的空间存储源字符串。toggle函数用于将字符串中的大小写转换,如果是小写字母则转换为大写,如果是大写字母则转换为小写。 在十进制转二进制...
将字符串逆序存放是指将一个字符串的元素逆序排列。例如,假设有一个字符串“hello”,那么将其逆序存放后变成了“olleh”。 15. 查找字符 查找字符是指在一个字符串中,查找某个特定的字符。例如,假设有一个字符...
1. **字符串逆序**:这是一个基本的字符串操作,通过创建一个指向目的地的指针和源字符串的指针,然后从源字符串的末尾开始,逐个字符复制到目标位置,直到字符串开头。在C++中,可以使用`std::reverse`函数实现,但...
这些问题涵盖了解析字节中被置位的数量、字符串到整数的转换、整数到字符串的转换、字符串逆序以及字符串中特定字符的查找。每种问题都有多种实现方法,关键在于理解位操作、字符串处理函数和基本的算法技巧。 ### ...
16. **字典树(Trie)**:高效地存储和查找字符串,尤其适用于拼写检查和自动补全功能。 17. **分布式缓存一致性hash算法**:通过一致性hash环和虚拟节点分配,确保缓存节点的添加或移除对整体影响最小。 18. **跳表...
知识点:字符串处理、算法设计 2. 多态性:该题目考察了面向对象编程的基础概念,多态性是指一个操作在不同的类中可以有不同的实现方式。 知识点:面向对象编程、多态性 3. 二叉树的前序、中序和后续序列:该题目...
最后一题涉及字符串操作,包括计算字符串长度和逆序字符串。这里展示了如何使用递归来实现字符串的逆序。 - **代码解析**: - 函数`strlen`通过遍历字符串直到遇到空字符`'\0'`来计算长度。 - 函数`revers`使用递...
此外,字符串处理函数如`strcpy()`用于复制字符串,`strcat()`用于连接字符串,`strcmp()`用于比较字符串,`strlen()`用于获取字符串长度。 电报加密问题可以通过字符数组实现,将每个字符向后移动一位,注意处理...
- 这是一个字符串处理问题,可以通过滑动窗口或KMP算法找到最长重复子串。 6. **斐波那契数列与质数**: - 斐波那契数列是一个特定的数列,质数斐波那契数是指与自身前面的斐波那契数互质的数。 - 找到第k小的...
### 算法参考手册实用知识点汇总 #### 几何篇 **1.1 注意事项** - **舍入方式:** 在处理浮点运算时,要注意0.5的舍入方向,确保结果的一致性和准确性。 - **测试数据:** 多测试不对称的数据可以发现算法中的...
在这篇文章中,我们将总结和分析100道超经典C语言编程题,涵盖了基本数据类型、算法、字符串处理、函数编程等多方面的知识点。 1. 求最大公约数和最小公倍数 在这个问题中,我们需要编写一个程序来计算两个正整数...