字符串类面试题型之一——字符串旋转
在笔试面试中,字符串类是经常重点考的题型。本系列是本人为了准备笔试与面试,综合各类书与博客,整理所得。
希望在此过程中,希望可以深刻理解各种题型,督促自己的学习进步,并且分享给大家。如有错误,非常希望得到指点,不甚感激。
题目描述:
定义字符串的左旋转操作:把字符串前面的若干个字符移动到字符串的尾部。如把字符串 abcdef 左旋转 2 位得到字符串 cdefab。
实现字符串左旋转的函数,要求对长度为 n 的字符串操作的时间复杂度为 O(n),空间复杂度为 O(1)。
本题来自于《编程之法》,书中给出了两种方法。第一种就是常见的,将需要移动的字符一个个的移动到字符串的尾部。但是此方法的时间复杂度是O(mn),空间复杂度为O(1).
第二种方法采用了递归的方式。本人认为是此类题目中最好的方法之一。特此推荐给大家。
实现代码如下:
void ReverseString(char *s, int from, int to)
{
char t;
while(from < to)
{
t = s[from];
s[from++] = s[to];
s[to--] = t;
}
}
void LeftRotateString_1(char *s, int n, int m)
{
m %= n;
ReverseString(s, 0, m-1);
ReverseString(s, m, n-1);
ReverseString(s, 0, n-1);
}
此方法一目了然,通过三步的反转每以此达到题目的要求。
由此拓展,采用此方法解决了其举一反三的题目:
输入一个英文句子,翻转句子中单词的顺序。要求单词内字符的顺序不变,句子中单词以空格符隔开,为简单起见,标点符号和普通字母一样处理。例如,若输入“I am student.”,则输出“student a am I”.
实现代码如下:
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
void ReverseString(char *s, int from, int to)
{
char t;
while(from < to)
{
t = s[from];
s[from++] = s[to];
s[to--] = t;
}
}
int main(int argc, char *argv[]) {
char s[100];
int len, i, m=0;
printf("Input String: ");
gets(s);
len = strlen(s);
for(i=0; i<len; i++)
{
if(s[i] == ' ')
{
ReverseString(s, m, i-1);
m = i+1;
}
else if(s[i] == '.')
{
ReverseString(s, m, i);
}
}
ReverseString(s, 0, len-1);
printf("Result: %s", s);
return 0;
}
这是我的第一篇博客,不足之处,敬请谅解。
对于题目与解法的来源,本人尽量给出出处,尊重原创,方便大家寻找。