字符串类面试题型之一——字符串旋转
字符串类面试题型之一——字符串旋转
在笔试面试中,字符串类是经常重点考的题型。本系列是本人为了准备笔试与面试,综合各类书与博客,整理所得。
希望在此过程中,希望可以深刻理解各种题型,督促自己的学习进步,并且分享给大家。如有错误,非常希望得到指点,不甚感激。
题目描述:
定义字符串的左旋转操作:把字符串前面的若干个字符移动到字符串的尾部。如把字符串 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; }
这是我的第一篇博客,不足之处,敬请谅解。
对于题目与解法的来源,本人尽量给出出处,尊重原创,方便大家寻找。