• 欢迎访问搞代码网站,推荐使用最新版火狐浏览器和Chrome浏览器访问本网站!
  • 如果您觉得本站非常有看点,那么赶紧使用Ctrl+D 收藏搞代码吧

关于PHP中strtr函数一些奇怪行为的解释

php 搞代码 4年前 (2022-01-22) 37次浏览 已收录 0个评论

这篇文章主要介绍了关于PHP中strtr函数一些奇怪行为的解释,有着一定的参考价值,现在分享给大家,有需要的朋友可以参考一下

PHP中strtr函数一些奇怪行为的解释

前些日子,一哥们给我发了篇文章给我看,说是strtr函数有一些奇怪的行为

查看PHP的源码,得到如下解释:

【奇怪行为一】
先来看看这个php字符串替换函数 strtr()的两种状态
strtr(string,from,to)
或者strtr(string,array)
首先针对strtr函数第一种方式
我们看看下面的举例:

   echo strtr("I Love you","Lo","lO");

得到的结果是
I lOve yOu

这个结果提醒我们

1.strtr它是区分大小写的

【源码分析一】
strtr函数的最终实现函数是string.c文件的2670行的php_strtr函数,其源码如下:

   PHPAPI char *php_strtr(char *str, int len, char *str_from, char *str_to, int trlen){int i;unsigned char xlat[256];  if ((trlen < 1) || (len < 1)) {return str;}  for (i = 0; i < 256; xlat[i] = i, i++);  for (i = 0; i < trlen; i++) {xlat[(unsigned char) str_from[i]] = str_to[i];}  for (i = 0; i < len; i++) {str[i] = xlat[(unsigned char) str[i]];}  return str;}

整个函数是对于256个字符进行hash替换,这256个字符中当然包括大小写字母啦

【奇怪行为二】

2.strtr的替换是很特殊的,你注意看后面那个yOu,中间的O被替换的,这显然不是我们的本意

【源码分析二】

道理同上,它是对每个字符进行对应替换,是以字符为单位,所以替换的是字符,而不是字符串

【奇怪行为三】

再举一个特殊例子,说明这个php sttr函数的怪异

   echo strtr("I Love you","Love","");

结果是

I Love you

什么也不会改变,所以strtr需要注意的是:不能被替换为空,也就是末位那个参数不能是空字符串,当然空格是可以的。

【源码分析三】

在string.c文件的2833行,我们可以看到其调用程序如下:

   php_strtr(Z_STRVAL_P(return_value), Z_STRLEN_P(return_value), Z_STRVAL_PP(from), Z_STRVAL_PP(to), MIN(Z_STRLEN_PP(from),  Z_STRLEN_PP(to)));

MIN(Z_STRLEN_PP(from), Z_STRLEN_PP(to))是取from和to两个字符串的长度中最小的,如三所示的情况,其结果为0,从php_strtr函数中我们可以看到

if ((trlen < 1) || (len < 1)) {return str;}

当长度小于1时返回原来的字符串。所以。。。。

【奇怪行为四】

再次举例strtr函数的另一种情况

 echo strtr("I Loves you","Love","lOvEA");

结果是

I lOvEs yOu

注意看第三个参数的A,在结果中并没有出现
【源码分析三】
其理由与二类似,MIN(Z_STRLEN_PP(from), Z_STRLEN_PP(to))是取from和to两个字符串的长度中最小的

  static void php_strtr_array(zval *return_value, char *str, int <a style="color:transparent">本@文来源gao($daima.com搞@代@#码(网5</a><strong>搞gaodaima代码</strong>slen, HashTable *hash){  zval **entry;char  *string_key;  uint   string_key_len;zval **trans;  zval   ctmp;ulong num_key;  int minlen = 128*1024;  int maxlen = 0, pos, len, found;  char *key;  HashPosition hpos;smart_str result = {0};  HashTable tmp_hash;zend_hash_init(&tmp_hash, zend_hash_num_elements(hash), NULL, NULL, 0);  zend_hash_internal_pointer_reset_ex(hash, &hpos);  while (zend_hash_get_current_data_ex(hash, (void **)&entry, &hpos) == SUCCESS) {  switch (zend_hash_get_current_key_ex(hash, &string_key, &string_key_len, &num_key, 0, &hpos)) {  case HASH_KEY_IS_STRING:len = string_key_len-1;  if (len < 1) {  zend_hash_destroy(&tmp_hash);  RETURN_FALSE;  }  zend_hash_add(&tmp_hash, string_key, string_key_len, entry, sizeof(zval*), NULL);  if (len > maxlen) {maxlen = len;}  if (len < minlen) {minlen = len;}break;   case HASH_KEY_IS_LONG:Z_TYPE(ctmp) = IS_LONG;Z_LVAL(ctmp) = num_key;  convert_to_string(&ctmp);len = Z_STRLEN(ctmp);  zend_hash_add(&tmp_hash, Z_STRVAL(ctmp), len+1, entry, sizeof(zval*), NULL);  zval_dtor(&ctmp);  if (len > maxlen) { maxlen = len;} if (len < minlen) { minlen = len;}break;} zend_hash_move_forward_ex(hash, &hpos);}  key = emalloc(maxlen+1);pos = 0;  while (pos < slen) { if ((pos + maxlen) > slen) {maxlen = slen - pos;}  found = 0;memcpy(key, str+pos, maxlen);  for (len = maxlen; len >= minlen; len--) { key[len] = 0; if (zend_hash_find(&tmp_hash, key, len+1, (void**)&trans) == SUCCESS) { char *tval; int tlen;zval tmp;  if (Z_TYPE_PP(trans) != IS_STRING) { tmp = **trans;zval_copy_ctor(&tmp);convert_to_string(&tmp); tval = Z_STRVAL(tmp);tlen = Z_STRLEN(tmp); } else {tval = Z_STRVAL_PP(trans); tlen = Z_STRLEN_PP(trans);}  smart_str_appendl(&result, tval, tlen);pos += len;found = 1;  if (Z_TYPE_PP(trans) != IS_STRING) {zval_dtor(&tmp);}break;} }  if (! found) {smart_str_appendc(&result, str[pos++]);}}  efree(key);zend_hash_destroy(&tmp_hash);smart_str_0(&result);RETVAL_STRINGL(result.c, result.len, 0);}

搞代码网(gaodaima.com)提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发送到邮箱[email protected],我们会在看到邮件的第一时间内为您处理,或直接联系QQ:872152909。本网站采用BY-NC-SA协议进行授权
转载请注明原文链接:关于PHP中strtr函数一些奇怪行为的解释

喜欢 (0)
[搞代码]
分享 (0)
发表我的评论
取消评论

表情 贴图 加粗 删除线 居中 斜体 签到

Hi,您需要填写昵称和邮箱!

  • 昵称 (必填)
  • 邮箱 (必填)
  • 网址