## 1.1 旋轉字符串
### 題目描述
給定一個字符串,要求把字符串前面的若干個字符移動到字符串的尾部,如把字符串“abcdef”前面的2個字符'a'和'b'移動到字符串的尾部,使得原字符串變成字符串“cdefab”。請寫一個函數完成此功能,要求對長度為n的字符串操作的時間復雜度為 O(n),空間復雜度為 O(1)。
### 分析與解法
#### 解法一:暴力移位法
初看此題,可能最先想到的方法是按照題目所要求的,把需要移動的字符一個一個地移動到字符串的尾部,如此我們可以實現一個函數`LeftShiftOne(char* s, int n)` ,以完成移動一個字符到字符串尾部的功能,代碼如下所示:
```c
void LeftShiftOne(char* s, int n)
{
char t = s[0]; //保存第一個字符
for (int i = 1; i < n; i++)
{
s[i - 1] = s[i];
}
s[n - 1] = t;
}
```
因此,若要把字符串開頭的m個字符移動到字符串的尾部,則可以如下操作:
```c
void LeftRotateString(char* s, int n, int m)
{
while (m--)
{
LeftShiftOne(s, n);
}
}
```
下面,我們來分析一下這種方法的時間復雜度和空間復雜度。
針對長度為n的字符串來說,假設需要移動m個字符到字符串的尾部,那么總共需要 m*n 次操作,同時設立一個變量保存第一個字符,如此,時間復雜度為O(m * n),空間復雜度為O(1),空間復雜度符合題目要求,但時間復雜度不符合,所以,我們得需要尋找其他更好的辦法來降低時間復雜度。
#### 解法二:三步反轉法
對于這個問題,換一個角度思考一下。
將一個字符串分成X和Y兩個部分,在每部分字符串上定義反轉操作,如X^T,即把X的所有字符反轉(如,X="abc",那么X^T="cba"),那么就得到下面的結論:(X^TY^T)^T=YX,顯然就解決了字符串的反轉問題。
例如,字符串 abcdef ,若要讓def翻轉到abc的前頭,只要按照下述3個步驟操作即可:
1. 首先將原字符串分為兩個部分,即X:abc,Y:def;
2. 將X反轉,X->X^T,即得:abc->cba;將Y反轉,Y->Y^T,即得:def->fed。
3. 反轉上述步驟得到的結果字符串X^TY^T,即反轉字符串cbafed的兩部分(cba和fed)給予反轉,cbafed得到defabc,形式化表示為(X^TY^T)^T=YX,這就實現了整個反轉。
如下圖所示:

代碼則可以這么寫:
```c
void ReverseString(char* s,int from,int to)
{
while (from < to)
{
char t = s[from];
s[from++] = s[to];
s[to--] = t;
}
}
void LeftRotateString(char* s,int n,int m)
{
m %= n; //若要左移動大于n位,那么和%n 是等價的
ReverseString(s, 0, m - 1); //反轉[0..m - 1],套用到上面舉的例子中,就是X->X^T,即 abc->cba
ReverseString(s, m, n - 1); //反轉[m..n - 1],例如Y->Y^T,即 def->fed
ReverseString(s, 0, n - 1); //反轉[0..n - 1],即如整個反轉,(X^TY^T)^T=YX,即 cbafed->defabc。
}
```
這就是把字符串分為兩個部分,先各自反轉再整體反轉的方法,時間復雜度為O(n),空間復雜度為O(1),達到了題目的要求。
### 舉一反三
1、鏈表翻轉。給出一個鏈表和一個數k,比如,鏈表為1→2→3→4→5→6,k=2,則翻轉后2→1→6→5→4→3,若k=3,翻轉后3→2→1→6→5→4,若k=4,翻轉后4→3→2→1→6→5,用程序實現。
2、編寫程序,在原字符串中把字符串尾部的m個字符移動到字符串的頭部,要求:長度為n的字符串操作時間復雜度為O(n),空間復雜度為O(1)。
例如,原字符串為”Ilovebaofeng”,m=7,輸出結果為:”baofengIlove”。
3、單詞翻轉。輸入一個英文句子,翻轉句子中單詞的順序,但單詞內字符的順序不變,句子中單詞以空格符隔開。為簡單起見,標點符號和普通字母一樣處理。例如,輸入“I am a student.”,則輸出“student. a am I”。
- 程序員如何準備面試中的算法
- 第一部分 數據結構
- 第一章 字符串
- 1.0 本章導讀
- 1.1 旋轉字符串
- 1.2 字符串包含
- 1.3 字符串轉換成整數
- 1.4 回文判斷
- 1.5 最長回文子串
- 1.6 字符串的全排列
- 1.10 本章習題
- 第二章 數組
- 2.0 本章導讀
- 2.1 尋找最小的 k 個數
- 2.2 尋找和為定值的兩個數
- 2.3 尋找和為定值的多個數
- 2.4 最大連續子數組和
- 2.5 跳臺階
- 2.6 奇偶排序
- 2.7 荷蘭國旗
- 2.8 矩陣相乘
- 2.9 完美洗牌
- 2.15 本章習題
- 第三章 樹
- 3.0 本章導讀
- 3.1 紅黑樹
- 3.2 B樹
- 3.3 最近公共祖先LCA
- 3.10 本章習題
- 第二部分 算法心得
- 第四章 查找匹配
- 4.1 有序數組的查找
- 4.2 行列遞增矩陣的查找
- 4.3 出現次數超過一半的數字
- 第五章 動態規劃
- 5.0 本章導讀
- 5.1 最大連續乘積子串
- 5.2 字符串編輯距離
- 5.3 格子取數
- 5.4 交替字符串
- 5.10 本章習題
- 第三部分 綜合演練
- 第六章 海量數據處理
- 6.0 本章導讀
- 6.1 關聯式容器
- 6.2 分而治之
- 6.3 simhash算法
- 6.4 外排序
- 6.5 MapReduce
- 6.6 多層劃分
- 6.7 Bitmap
- 6.8 Bloom filter
- 6.9 Trie樹
- 6.10 數據庫
- 6.11 倒排索引
- 6.15 本章習題
- 第七章 機器學習
- 7.1 K 近鄰算法
- 7.2 支持向量機
- 附錄 更多題型
- 附錄A 語言基礎
- 附錄B 概率統計
- 附錄C 智力邏輯
- 附錄D 系統設計
- 附錄E 操作系統
- 附錄F 網絡協議
- sift算法
- sift算法的編譯與實現
- 教你一步一步用c語言實現sift算法、上
- 教你一步一步用c語言實現sift算法、下
- 其它
- 40億個數中快速查找
- hash表算法
- 一致性哈希算法
- 倒排索引關鍵詞不重復Hash編碼
- 傅里葉變換算法、上
- 傅里葉變換算法、下
- 后綴樹
- 基于給定的文檔生成倒排索引的編碼與實踐
- 搜索關鍵詞智能提示suggestion
- 最小操作數
- 最短摘要的生成
- 最長公共子序列
- 木塊砌墻原稿
- 附近地點搜索
- 隨機取出其中之一元素