php在操作字串的問題時間無非兩個問題:
1.判斷字串編碼是gbk還是unicode。
2.對相應編碼採取相應截取方法。
一般情況下我們使用substr截取漢字可能會遇到亂碼問題。因為漢字是雙位元組的,當被截取了一個位元組時,這個漢字就無法顯示,亂掉了。
其實解決很簡單,看下面的截取函數:
| 代碼如下 |
複製代碼 |
//截取超長字串 function curtStr($str,$len=30){ if(strlen($str)>$len){ $str = substr($str,0,$len); $str .= chr(0) ."…"; return $str; } |
上面的chr(0)不是null
null是什麼都沒有,而chr(0)的值是0。表示成16進位是0×00,表示成二進位是00000000
雖然chr(0)不會顯示出什麼,但是他是一個字元。
當漢字被截斷時,根據編碼規則他總是要把後邊的其他字元拉過來一起作為漢字解釋,這就是出現亂碼的原因。而值為0×81到0xff與0×00組合始終都顯示為“空”
根據這一特點,在substr的結果後面補上一個chr(0),就可以防止出現亂碼了
下面補充幾個函數即可實現此兩點以達到精確截取中文字串的目的:
截取utf8編碼的多位元組字串
| 代碼如下 |
複製代碼 |
<?php //截取utf8字串 function utf8Substr($str, $from, $len) { return preg_replace('#^(?:[x00-x7F]|[xC0-xFF][x80-xBF]+){0,'.$from.'}'. '((?:[x00-x7F]|[xC0-xFF][x80-xBF]+){0,'.$len.'}).*#s', '$1',$str); } ?> |
UTF-8、GB2312都支援的漢字截取函數
| 代碼如下 |
複製代碼 |
<?php /* Utf-8、gb2312都支援的漢字截取函數 cut_str(字串, 截取長度, 開始長度, 編碼); 編碼預設為 utf-8 開始長度預設為 0 */ function cut_str($string, $sublen, $start = 0, $code = 'UTF-8') { if($code == 'UTF-8') { $pa = "/[x01-x7f]|[xc2-xdf][x80-xbf]|xe0[xa0-xbf][x80-xbf]|[xe1-xef][x80-xbf][x80-xbf]|xf0[x90-xbf][x80-xbf][x80-xbf]|[xf1-xf7][x80-xbf][x80-xbf][x80-xbf]/"; preg_match_all($pa, $string, $t_string); if(count($t_string[0]) - $start > $sublen) return join('', array_slice($t_string[0], $start, $sublen))."..."; return join('', array_slice($t_string[0], $start, $sublen)); } else { $start = $start*2; $sublen = $sublen*2; $strlen = strlen($string); $tmpstr = ''; for($i=0; $i<$strlen; $i++) { if($i>=$start && $i<($start+$sublen)) { if(ord(substr($string, $i, 1))>129) { $tmpstr.= substr($string, $i, 2); } else { $tmpstr.= substr($string, $i, 1); } } if(ord(substr($string, $i, 1))>129) $i++; } if(strlen($tmpstr)<$strlen ) $tmpstr.= "..."; return $tmpstr; } } $str = "abcd需要截取的字串"; echo cut_str($str, 8, 0, 'gb2312'); ?> |