亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > iOS > 正文

詳細談談iOS字符串翻轉

2020-07-26 02:25:10
字體:
來源:轉載
供稿:網友

前言

字符串翻轉作為算法題已經是一個不能再基礎的問題了,無非就是逆序遍歷、雙指針遍歷、遞歸,代碼也能分分鐘寫出來:

void strrev(char *str) { size_t start = 0; size_t end = start + strlen(str) - 1; while (start < end) {  char ch = str[start];  str[start++] = str[end];  str[end--] = ch; }}

OK,上面的代碼放到 LeetCode 上絕對是能 AC 的,但是實際情況中能 AC 嗎?答案肯定是不能的!一個靠譜的字符串翻轉算法題放到 LeetCode 上至少是 Medium 的難度。

首先我們知道字符串有編碼規則,比如我們常用的 UTF-8,Windows 早期采用的 UTF-16(函數名有 W 后綴的 API 采用這種編碼)等等...對于英文字母等 ASCII 字符的情況,UTF-8 和 ASCII 編碼都是一個字節,所以上述的方法沒有太大問題。然而對于有中文的情況,一個中文字符在 UTF-8 中會占 3 個字節,如果單純的按字節翻轉就會出現亂碼。

那怎么解決呢?

最簡單的方法就是使用 mbstowcs 函數將 char * 類型的字符串轉換為 wchar_t 類型的寬字符串,wchar_t 這個類型在 Linux、UNIX 系統上占 4 個字節,在 Windows 上占 2 個字節。4 個字節意味著字符將用 UTF-32 來編碼,不管是漢字還是 Emoji 都能存放下來。但對于 2 個字節,也就是 UTF-16,漢字是能表示,但是 Emoji 這類位于輔助平面碼位的字符需要兩個碼元來表示,本文的方法就暫不適用了。

首先我們來看一下改進版的字符串翻轉:

static void strrev2(char *str) { setlocale(LC_CTYPE, "UTF-8"); size_t len = mbstowcs(NULL, str, 0); wchar_t *wcs = (wchar_t *) calloc(len + 1, sizeof(wchar_t)); mbstowcs(wcs, str, len + 1); size_t start = 0; size_t end = start + len - 1; while (start < end) {  wchar_t wc = wcs[start];  wcs[start++] = wcs[end];  wcs[end--] = wc; } wcstombs(str, wcs, wcstombs(NULL, wcs, 0)); free(wcs);}

使用 mbstowcs 這類轉換函數首先需要設置字符串的系統編碼,不然函數無法確定你傳入的 char * 是個什么東西,本文中不管是源碼還是系統環境的 std I/O 都采用 UTF-8 編碼。

接下來我們調用一次 mbstowcs 不傳入目標地址和字符長度,這可以讓函數直接計算所需的 wchar_t 個數并返回回來以便我們申請內存。

然后就是基于 wchar_t 的一個常規字符串翻轉了,最后別忘了轉換回去,釋放內存即可。

Bonus: Cocoa 開發中的字符串翻轉

作為 iOS 開發者,當然還要考慮 OC 中的解決方法了。

方案 1:

通過 API 遍歷子串,然后前向插入到新的 NSMutableString 中。

- (NSString *)my_stringByReversing { NSMutableString *reversed = [NSMutableString stringWithCapacity:self.length]; NSRange range = NSMakeRange(0, self.length); [self enumerateSubstringsInRange:range        options:NSStringEnumerationByComposedCharacterSequences       usingBlock:^(NSString * _Nullable substring, NSRange substringRange,          NSRange enclosingRange, BOOL * _Nonnull stop) {        [reversed insertString:substring atIndex:0];       }]; return [reversed copy];}

這種方法是效果最好的,它會將 Composed Emoji(如 👨‍👩‍👧‍👧)也提取出來,因為這類 Emoji 是由多個 Unicode 字符組合而成的,所以即便是 4 個字節的 wchar_t 也容納不下。但這種方法的弊端就是開銷太大,稍后我們做一個比較。

方案 2:

通過 API 獲取到 C String,然后用文章開頭所述的方法處理,再重新用處理后的 C String 構造 NSString。

- (NSString *)my_stringByReversing2 { NSUInteger length = [self lengthOfBytesUsingEncoding:NSUTF8StringEncoding]; char *buf = calloc(length + 1, 1); [self getCString:buf maxLength:length + 1 encoding:NSUTF8StringEncoding]; strrev2(buf); NSString *reversed = [NSString stringWithCString:buf encoding:NSUTF8StringEncoding]; free(buf); return reversed;}

這種方法的好處就是高效,經測試,它與遍歷的方法相比有 100 多倍的性能提升,但是問題就是無法處理復雜的 Emoji。

兩種方法,在使用中需要好好衡量一下。

方案 3:

Swift。Swift 的 String 的基本單位是 Character,它是 Unicode Scalar 的集合,表示了一個可渲染的字符,包括 Composed Emoji。并且,String 是實現了 BidirectionalCollection,擁有 reversed 方法,可以輕松實現字符串翻轉。另外要提醒大家一下,正由于 Swift 的 String 是基于 Character 的,對于取某個字符這樣的操作,能復用之前的 Index 就復用,我見過很多人喜歡寫

str.index(str.startIndex, offsetBy: i)

這樣是很費性能的,因為 Index 的移動操作需要從起點遍歷計算,用這種方法遍歷一遍字符串的復雜度近似是 O(n!)。

大家有興趣可以試試 Swift 的性能~

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對武林網的支持。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
久久中文精品视频| 国产精品xxxxx| 国产精品视频免费在线观看| 中文字幕在线观看亚洲| 亚洲国产古装精品网站| 国产成人一区二区| 久久人人爽亚洲精品天堂| 国产91精品不卡视频| 精品香蕉一区二区三区| 亚洲人成电影网站色xx| 亚洲天堂日韩电影| 亚洲欧美中文在线视频| 国产精品扒开腿做爽爽爽视频| 国产日韩欧美91| 欧美日本高清视频| 成人深夜直播免费观看| 97热精品视频官网| 午夜精品美女自拍福到在线| 成人精品在线观看| 亚洲va男人天堂| 91最新国产视频| 亚洲精品按摩视频| 91久久久久久国产精品| 亚洲国产婷婷香蕉久久久久久| 亚洲国产第一页| 精品国产999| 亚洲欧美一区二区精品久久久| 久久精品最新地址| 精品高清一区二区三区| 亚洲伊人久久大香线蕉av| 欧美日韩国产精品一区二区不卡中文| 38少妇精品导航| 亚洲欧洲日产国码av系列天堂| 日本老师69xxx| 亚洲已满18点击进入在线看片| 欧美又大又硬又粗bbbbb| 久久久久九九九九| 亚洲a级在线播放观看| 97超级碰碰碰久久久| 国产盗摄xxxx视频xxx69| 亚洲精品国产福利| 人人澡人人澡人人看欧美| 国产一区二区三区视频免费| 亚洲一区二区三区在线视频| 91沈先生作品| 成人xvideos免费视频| 亚洲欧美国产一本综合首页| 精品国产一区二区三区久久久狼| 精品福利免费观看| 日韩精品在线视频观看| 亚洲一级黄色片| 久久久久久国产免费| 国产欧美日韩精品丝袜高跟鞋| 亚洲黄色www| 色婷婷**av毛片一区| 国内精品久久久久影院 日本资源| 成人欧美一区二区三区黑人| 国产精品久久久久久久久久久久| 国产精品国产亚洲伊人久久| 亚洲国产天堂久久综合网| 国产日韩精品综合网站| 欧美激情一区二区三区成人| 亚洲精品国产拍免费91在线| 在线亚洲男人天堂| 亚洲最新av在线| 国产欧美日韩专区发布| 欧美一区二区三区精品电影| 亚洲欧美999| 欧美一区第一页| 久久久国产精品一区| 中文字幕久久久| 欧美亚洲日本网站| 日韩av综合中文字幕| 久久久成人的性感天堂| 日韩av毛片网| 97精品国产91久久久久久| 国产精品嫩草视频| 精品国产区一区二区三区在线观看| 亚洲奶大毛多的老太婆| 在线色欧美三级视频| 亚洲国产精品久久91精品| 国内精品美女av在线播放| 日韩av在线网| 国产精品h片在线播放| 欧美尺度大的性做爰视频| 亚洲男人第一av网站| 在线不卡国产精品| 久久久在线视频| 久久精品国产91精品亚洲| 亚洲国产精品嫩草影院久久| 欧美人交a欧美精品| 亚洲国产精品嫩草影院久久| 国产一区二区三区免费视频| 欧洲日韩成人av| 国产精品亚洲网站| 久久久久久噜噜噜久久久精品| 这里只有精品久久| 欧美精品制服第一页| 久久影院免费观看| 午夜精品蜜臀一区二区三区免费| 日韩精品免费综合视频在线播放| 欧美激情视频一区二区| 欧美综合一区第一页| 视频在线观看99| 国产啪精品视频网站| 国产精品人成电影| 日韩黄色高清视频| 91九色单男在线观看| 欧美性极品少妇精品网站| 怡红院精品视频| 亚洲国产成人精品电影| 欧美日韩免费区域视频在线观看| 久久精品亚洲94久久精品| 欲色天天网综合久久| 欧美激情视频免费观看| 尤物精品国产第一福利三区| 久久久久久91香蕉国产| 国产日本欧美一区二区三区在线| 欧美在线日韩在线| 日韩第一页在线| 国产精品扒开腿做爽爽爽的视频| 91在线视频导航| 97久久久久久| 精品国产一区二区三区久久狼黑人| 久久久久久久久爱| 亚洲少妇激情视频| 精品美女久久久久久免费| 欧美专区中文字幕| 久久久久久久久久久国产| 亚洲第一级黄色片| 97国产一区二区精品久久呦| 午夜精品视频在线| 久久久久久久久中文字幕| 色综合视频一区中文字幕| 国产精品爽爽爽爽爽爽在线观看| 国产欧美精品日韩精品| 两个人的视频www国产精品| 一个色综合导航| 欧美极品美女电影一区| 成人免费网站在线| 在线不卡国产精品| 国产亚洲精品91在线| 国产精品三级美女白浆呻吟| 欧美日韩亚洲网| 久久久久久久久久婷婷| 91精品在线播放| 日韩欧美极品在线观看| 精品一区二区三区电影| 97视频免费观看| 2018国产精品视频| 久久精品国产2020观看福利| 亚洲成年人在线播放| 欧日韩在线观看| 欧美成人网在线| 欧美成aaa人片在线观看蜜臀| 国产精品直播网红| 清纯唯美日韩制服另类| 在线日韩日本国产亚洲| 国产在线播放91| 欧美视频在线视频| 欧美裸体男粗大视频在线观看| 久久精品国产96久久久香蕉| 欧美日韩国产综合视频在线观看中文| 国产亚洲综合久久|