關於php trim方法的錯誤理解導致的問題
場景
php中的擷取字串前後字元包括有:ltrim
,rtrim
,trim
三個方法
下面的例子中只以ltrim方法做舉例
在我之前的認知中(當然我很水,從沒看過這塊原始碼),如果我想要刪除字串左邊的空字串,空製表符之類的,那麼我就直接使用ltrim($str)
即可
如果我想要刪除指定字元的時候,比如說現在有個字串helloworld,我要擷取掉頭部的h字元,直接var_dump(ltrim("helloworld", "h"));
即可得到我期望的結果輸出elloworld
以上的都是在我以為的範圍內,我也一直都是這麼使用的,直到有一次我們有個需求要在一些字串上做openssl_encrypt
加密,加密之後做個base64,然後拼接上我們的特殊的字串字首KO:
,每次加密完成後拼接KO:
字元,同樣的,解密之前先把KO:
拆出去在解密,結果發現解密怎麼解都是失敗,後來打了幾個斷點發現是ltrim的時候和預期結果不一樣
復現
<?php // 以下做一個簡單的情景復現 $str = "abccabdefg"; $character_mask = “abc”; var_dump(ltrim($str,$character_mask));// 輸出結果為 bdefg,而不是隻擷取前三位的abc
原因分析
經過上面的小demo,大家應該就知道原因是啥了,說的最簡單通俗的就是它把前面的$str
做一個輪訓,一個字元一個字元的在後面的$character_mask
裡面看是不是在其中,如果是的話則進行擷取,不在的話停止執行
ltrim程式碼形式的表達:
<?php $str = "abcccabdecbag"; $res = ''; $character_mask = 'abc'; $arrStr = str_split($str);// 字串轉陣列 for ($i=0; $i<=count($arrStr); $i++) { if(strpos($character_mask, $arrStr[$i]) !== false) { unset($arrStr[$i]); } else { break; } } echo ltrim($str, $character_mask); echo "\r\n"; echo implode($arrStr); echo "\r\n";
解決方案
解決方案就是使用php中的一些操作字串函式,多加了基層判斷
if (substr($str, 0, strlen($character_mask)) == $character_mask) { echo substr($str, strlen($character_mask)); }