如果是简单的字符截取我们常用的就substr函数或再使用mb_substr来截取字符了,但是我们有时会发现有中英文混排字符串截取时并不像那么简单,因这样我们需要考虑到编码问题,如ascii、16进制、正则匹配、循环计数,但今天文章介绍的不需要想到这些用到一个我们不常用的字符截取函数,mb_strwidth这个是php自带的函数.
mb_strwidth($str, $encoding) 返回字符串的宽度
$str 要计算的字符串
$encoding 要使用的编码,如 utf8、gbk
mb_strimwidth($str,$start,$width,$tail,$encoding) 按宽度截取字符串
$str 要截取的字符串
$start 从哪个位置开始截取,默认是0
$width 要截取的宽度
$tail 追加到截取字符串后边的字符串,常用的是 ...
$encoding 要使用的编码
实例代码如下:
- <?php
-
-
-
-
-
-
-
-
- $str = 'aaaa啊啊aaaa啊啊啊aaa';
- echo strlen($str);
-
-
-
- echo mb_strwidth($str, 'utf8');
-
-
- if(mb_strwidth($str, 'utf8')>10){
-
-
- $str = mb_strimwidth($str, 0, 10, '...', 'utf8');
- }
-
-
-
-
- echo $str;
- ?>
如果对于全中文没有问题但如果中间有符号了就有问题了,如我使用mb_strimwidth,mb_strwidth,后发现如果标题中存在“”符号的时候,PHP mb_strwidth会将该符号认为是1个宽度,我纳闷了这个不是中文的双引号嘛,照理说肯定是宽字节的,长度应该是2个宽度,后查询“”unicode分别为u201C和u201D,不在中文字符的范围中,再查询unicode.org 的码表,发现u2000-u206F是通用符号的范围,此范围中的字符虽然都是宽字符的形式,但是PHP 的mb_函数却认为是1个宽度,没办法,只能靠自己了,代码如下:
- function truncString($str, $length)
- {
- $countLen=0;
- for($i=0;$i<mb_strlen($str);$i++)
- {
- $countLen+=amb_strwidth(mb_substr($str,$i,1));
- if($countLen>$length)
- return mb_substr($str,0,$i);
- }
- return $str;
- }
- function amb_strwidth($str_width)
- {
- $count=0;
- for($i=0;$i<mb_strlen($str_width);$i++)
- {
-
-
- if(preg_match("/[\x{2000}-\x{206F}]/u",mb_substr($str_width,$i,1)))
- $count+=2;
- else
- $count+=mb_strwidth(mb_substr($str_width,$i,1));
- }
- return $count;
- }
总结:做来做出怎么就感觉到这个变成了回原点了呢,感觉还是要使用到循环遍历算字符编码取字符位数. |