移動前端手機輸入法自帶emoji表情字符處理


 今天,測試給我提了一個BUG,說移動端輸入emoji表情無法提交。很早以前就有思考過,手機輸入法里自帶的emoji表情,應該是某些特殊字符。既然是字符,那應該都能提交才對,可是為啥會被卡住呢?搜了一下,才發現,原來emoji用到的字符是4字節的utf-16(utf-16有2字節和4字節兩種編碼),而我們的數據庫是采用的utf-8,並且最大只允許3字節的字符。這樣沖突就產生了,表單因為這些emoji字符的存在無法提交。

  找到原因之后,接下來就要考慮解決方案了。目前考慮到的兩種方案,一是讓后台處理,把這個utf-16字符做一些轉換(這里不做討論)。第二種辦法就是在前端直接轉換成實體字符后再提交。這樣,后台不用做任何處理,用戶的提交的信息也得以保留,是不是一個兩全其美的辦法呢?大笑接下來我們要討論的就是怎么把emoji表情字符轉換成實體字符。

  首先,我們來看看手機輸入法里自帶的emoji字符是什么樣。下面截了一張圖,來自 http://computerism.ru/emoji-smiles.htm 。我們看到,每個emoji表情字符對應的實體字符編碼都比較大,如第一行的笑臉,實體字符為😊 。而且,我們注意到,后面還有一個16進制的編碼 D83DDE0A。那這個編碼是干嘛用的?接着往下看。

 

一、字符檢測

  要想把這些emoji表情字符轉換成實體字符,那么就要先把它們檢測出來。說到字符檢測,我們的正則這時就該上場了。首先我們得確定這些字符的范圍。前面我們已經知道,emoji表情字符用的是4字節的utf-16編碼,而4字節的utf-16編碼不被后台接受。所以,我們的檢測范圍就變成了把所有4字節的utf-16編碼檢測出來。我們通過搜索查到,4字節的utf-16編碼范圍為U+010000到U+10FFFF,那么,我們的正則是不是可以這么寫:/[\u010000-\u10FFFF]/g ? NO,你會發現這個正則完全不能按我們預期工作。這是為什么呢?

  上面這個問題,一些童鞋可能已經知道答案了。沒錯,就是javascript的編碼問題引起的。我們知道,javascript采用的是unicode編碼,再准確一點說,是ucs-2編碼。從名字上,我們就已經知道,這種編碼方案是2字節的。在2字節的編碼中找4字節的字符,很顯然並沒那么簡單。所以,我們得考慮一下,這個utf-16在ucs-2編碼中是如何表示的呢?這里,我搜到了我們可愛的傳教士——阮老師的一篇文章 《Unicode與JavaScript詳解》(http://www.ruanyifeng.com/blog/2014/12/unicode.html) 。 簡單來說,就是把utf-16的4字節字符,拆分成兩個ucs-2的2字節字符。具體算法可參考阮老師的上述文章,本文就不詳細討論了。從阮老師的文章中,我們已經知道了,4字節utf-16在js中被用兩個字符來表示,高位范圍為0xD800 - 0xDBFF,低位范圍為0xDC00 - 0xDFFF。那么我們用於檢測的正則表達式也就出來了:/[\uD800-\uDBFF][\uDC00-\uDFFF]/g 。現在再回過頭看看我們第一張圖的那串16進制,D83DDE0A、D83DDE03,是不是突然就明白了呢?大笑

 

二、轉換算法

  現在,我們已經能夠檢測出表單里的emoji表情字符。那么,重頭戲來了,我們怎么把這個字符轉換成實體字符呢?我們知道,實體字符是用來表示單個字符的編碼,而我們的emoji表情,在js里,卻是用兩個字符來表示的。這可怎么辦?等等,誰說emoji是兩個字符,說好的4字節單字符呢?沒錯,一開始emoji就是用utf-16表示的啊尷尬 這里,我又參考了另一篇文章,http://unicode-table.com/cn/sets/emoji/,以下截了一部分圖以做說明。

  我們還是以那個笑臉的字符為例,其utf-16編碼為U+1F600,我們轉成十進制看看。

  128512不正好是我們的實體編碼😀 嗎?所以,現在問題又變成了怎么取得emoji表情字符utf-16編碼的問題了。可是,可是,我們剛剛已經知道了,在js里,emoji表情也是用ucs-2編碼的啊,只不過變成了用兩個字符來表示。那么,我們的問題最終演變成了怎么從ucs-2編碼轉換成utf-16編碼的問題。

  感謝阮老師,在阮老師的那篇文章中,有提到utf-16轉ucs-2(unicode)的公式

 

[javascript]  view plain  copy
 
  1. H = Math.floor((c-0x10000) / 0x400)+0xD800 // 高位  
  2.   
  3. L = (c - 0x10000) % 0x400 + 0xDC00 // 低位  

  可是,這個是utf-16轉ucs-2,我們要的是ucs-2轉utf-16啊?怎么辦?推導回去唄。我們先看看這兩個公式都做了什么。首先,高位的公式,把字符C減0x10000,再除0x400,取其商,再加0xD800。而低位則是字符C減0x1000,取除0x400的余,再加0xDC00。所以這個字符其實被分成了兩部分:商和余,然后再把處理后的商做為高位,加上處理后的余做為低位,這樣組合成了ucs-2字符。我們知道,被除數=除數×商+余數。那么,我們也可以反過來,求得C/0x400的商,再加上C/0x400的余,不就能算出C了嗎。為了便於計算,我們用Q表示C的商,用M表示C的余,那么就有了以下公式:

 

 

[javascript]  view plain  copy
 
  1. H = Q - 0x10000 / 0x400 + 0xD800  
  2. L = M - 0x10000 % 0x400 + 0xDC00  
  3. C = Q * 0x400 + M  
  4. // 因為0x10000 % 0x400 = 0,故推得:  
  5. H = Q - 0x10000 / 0x400 + 0xD800  
  6. L = M + 0xDC00  
  7. C = Q * 0x400 + M  
  8. // 根據C的公式,把H*0x400再加L,得到:  
  9. H * 0x400 + L = Q * 0x400 - 0x10000 / 0x400 * 0x400 + 0xD800 * 0x400 + M + 0xDC00  
  10. // 最后把Q * 0x400 + M換成C,得到:  
  11. H * 0x400 + L = C - 0x10000 + 0xD800 * 0x400 + 0xDC00  
  12. // 移項后,我們最終的公式為:  
  13. C = (H - 0xD800) * 0x400 + 0x10000 + L - 0xDC00  

   公式出來之后,相信大家已經知道怎么做了,不過最后還是獻一下丑,把我自己寫的一個處理函數提供給大家參考:

 

[javascript]  view plain  copy
 
  1. /** 
  2.  * 用於把用utf16編碼的字符轉換成實體字符,以供后台存儲 
  3.  * @param  {string} str 將要轉換的字符串,其中含有utf16字符將被自動檢出 
  4.  * @return {string}     轉換后的字符串,utf16字符將被轉換成&#xxxx;形式的實體字符 
  5.  */  
  6. function utf16toEntities(str) {  
  7.     var patt=/[\ud800-\udbff][\udc00-\udfff]/g; // 檢測utf16字符正則  
  8.     str = str.replace(patt, function(char){  
  9.             var H, L, code;  
  10.             if (char.length===2) {  
  11.                 H = char.charCodeAt(0); // 取出高位  
  12.                 L = char.charCodeAt(1); // 取出低位  
  13.                 code = (H - 0xD800) * 0x400 + 0x10000 + L - 0xDC00; // 轉換算法  
  14.                 return "&#" + code + ";";  
  15.             } else {  
  16.                 return char;  
  17.             }  
  18.         });  
  19.     return str;  
  20. }  

 

   運行結果如下:

 

utf16toEntities(unescape(%uD83D%udE0A));

  細心的童鞋,在剛剛看那些參考文章的時候,也許已經發現了,其實並不是所有的emoji表情字符都是utf-16編碼的,也有一部分落在了ucs-2編碼的范圍(即只用了兩個字節)。不過這都不是重點,重點是,我們已經成功的把utf-16編碼部分的emoji表情轉換為了實體字符。

 

轉:http://blog.csdn.net/binjly/article/details/47321043

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM