0

0

深入PHP内核之in_array,php内核in_array_PHP教程

php中文网

php中文网

发布时间:2016-07-12 09:05:21

|

1339人浏览过

|

来源于php中文网

原创

深入PHP内核之in_array,php内核in_array

无意中看到一段代码

1、a.php

<?php
$y="12";
$x = array();
for($j=0;$j<50000;$j++){
    $x[]= "{$j}";
}

for($i=0;$i<30000;$i++){
    if(in_array($y,$x,true)){
        continue;
    }
}
?>

测试

立即学习PHP免费学习笔记(深入)”;

[root@dev tmp]# time php a.php

real    0m0.101s
user    0m0.080s
sys     0m0.013s

2、b.php

<?php                                                                                                                                                                                        
$y="1800";
$x = array();
for($j=0;$j<50000;$j++){
    $x[]= "{$j}";
}
 
for($i=0;$i<30000;$i++){
    if(in_array($y,$x)){
        continue;
    }   
}

测试

立即学习PHP免费学习笔记(深入)”;

[root@dev tmp]# time php b.php 

real    0m9.517s
user    0m4.486s
sys     0m0.015s

需要9s

 

对于b.php 有严重的效率问题,跟踪测试一下好了

[root@dev tmp]# ltrace -c  php b.php     
% time     seconds  usecs/call     calls      function
------ ----------- ----------- --------- --------------------
 52.92  175.486683         224    780155 strtol
 33.34  110.550182    55275091         2 __libc_start_main
  6.33   20.999979         177    118479 memcpy
  3.56   11.819558         234     50394 __ctype_b_loc
  1.66    5.510564         285     19294 free
  1.42    4.704605      156820        30 dlopen
  0.43    1.416750          76     18430 malloc
  0.13    0.422050          76      5510 strlen
  0.08    0.258620          76      3375 __ctype_tolower_loc
  0.04    0.137245          77      1770 strrchr
  0.02    0.067158          76       880 strcasecmp
  0.02    0.059222          76       776 calloc
  0.01    0.034591       17295         2 getprotobyname
  0.01    0.031099          76       407 realloc
  0.01    0.021277          77       276 memset
  0.00    0.011502         383        30 dlclose
  0.00    0.010671          85       125 
  0.00    0.006806          73        93 fileno
  0.00    0.006095          76        80 strncasecmp
  0.00    0.005397          77        70 strchr
  0.00    0.005139         160        32 ftell
  0.00    0.004405         137        32 fclose
  0.00    0.003644         104        35 __fxstat
  0.00    0.003432         107        32 fopen
  0.00    0.003103         107        29 munmap
  0.00    0.003062          78        39 getenv
  0.00    0.003031         104        29 mmap
  0.00    0.003027         104        29 isatty
  0.00    0.002825         104        27 __xstat
  0.00    0.002468          82        30 dlsym
  0.00    0.002167          74        29 sysconf
  0.00    0.001956          75        26 _setjmp
  0.00    0.001419         109        13 __lxstat
  0.00    0.001294          76        17 memchr
  0.00    0.001125        1125         1 SYS_clone
  0.00    0.001099        1099         1 tzset
  0.00    0.001098        1098         1 exit
  0.00    0.001075        1075         1 ERR_load_crypto_strings
  0.00    0.000986         986         1 using_history
  0.00    0.000837         837         1 SYS_exit_group
  0.00    0.000606          75         8 __strdup
  0.00    0.000596          74         8 strcmp
  0.00    0.000591          73         8 __memcpy_chk
  0.00    0.000457          76         6 fflush
  0.00    0.000326         108         3 getcwd
  0.00    0.000309          77         4 __errno_location
  0.00    0.000271         271         1 setlocale
  0.00    0.000271         271         1 scandir
  0.00    0.000238          79         3 __sprintf_chk
  0.00    0.000222         111         2 signal
  0.00    0.000206         206         1 SSL_library_init
  0.00    0.000186         186         1 fgetc
  0.00    0.000155         155         1 ERR_load_ERR_strings
  0.00    0.000154          77         2 xmlSetGenericErrorFunc
  0.00    0.000151          75         2 __strtok_r
  0.00    0.000151          75         2 xmlParserInputBufferCreateFilenameDefault
  0.00    0.000148         148         1 xmlInitParser
  0.00    0.000147          73         2 xmlOutputBufferCreateFilenameDefault
  0.00    0.000130         130         1 OpenSSL_add_all_ciphers
  0.00    0.000122         122         1 EVP_cleanup
  0.00    0.000112         112         1 access
  0.00    0.000109         109         1 rewind
  0.00    0.000106         106         1 OPENSSL_add_all_algorithms_noconf
  0.00    0.000105         105         1 sigprocmask
  0.00    0.000093          93         1 gnu_get_libc_version
  0.00    0.000090          90         1 OpenSSL_add_all_digests
  0.00    0.000084          84         1 xmlCleanupParser
  0.00    0.000081          81         1 xmlRelaxNGCleanupTypes
  0.00    0.000079          79         1 xmlSetStructuredErrorFunc
  0.00    0.000078          78         1 SSL_get_ex_new_index
  0.00    0.000077          77         1 __gmp_set_memory_functions
  0.00    0.000076          76         1 ERR_load_EVP_strings
  0.00    0.000076          76         1 pcre_version
  0.00    0.000075          75         1 X509_get_default_cert_area
  0.00    0.000075          75         1 strstr
  0.00    0.000074          74         1 sigemptyset
  0.00    0.000074          74         1 __xmlParserVersion
  0.00    0.000074          74         1 time
  0.00    0.000074          74         1 xmlResetLastError
  0.00    0.000074          74         1 strncmp
  0.00    0.000073          73         1 sigaddset
------ ----------- ----------- --------- --------------------
100.00  331.614442               1000662 total

我们发现  strtol 占用了大量的时间

查一下库函数

/*
函数名: strtol 
功  能: 将串转换为长整数 
用  法: long strtol(char *str, char **endptr, int base); 
程序例: 
*/

#include <stdlib.h> 
#include <stdio.h> 

int main(void) 
{ 
   char *string = "87654321", *endptr; 
   long lnumber; 

   /* strtol converts string to long integer  */ 
   lnumber = strtol(string, &endptr, 10); 
   printf("string = %s  long = %ld\n", string, lnumber); 

   return 0; 
} 

所以应该是源代码中有大量的类型转换

 

关于in_array

in_array是这个样子的

bool in_array ( mixed $needle , array $haystack [, bool $strict = FALSE ] )

在 haystack 中搜索 needle,如果没有设置 strict 则使用宽松的比较。

needle

待搜索的值。如果 needle 是字符串,则比较是区分大小写的。

haystack

这个数组。

strict

PHP与MySQL程序设计3
PHP与MySQL程序设计3

本书是全面讲述PHP与MySQL的经典之作,书中不但全面介绍了两种技术的核心特性,还讲解了如何高效地结合这两种技术构建健壮的数据驱动的应用程序。本书涵盖了两种技术新版本中出现的最新特性,书中大量实际的示例和深入的分析均来自于作者在这方面多年的专业经验,可用于解决开发者在实际中所面临的各种挑战。 本书内容全面深入,适合各层次PHP和MySQL开发人员阅读,既是优秀的学习教程,也可用作参考手册。

下载

如果第三个参数 strict 的值为 TRUE 则 in_array() 函数还会检查 needle 的类型是否和 haystack 中的相同。

那么我看一下源代码

第一步 在ext/standard/array.c 文件中

/* }}} */                                                      
                                                               
/* {{{ proto bool in_array(mixed needle, array haystack [, bool strict])
   Checks if the given value exists in the array */            
PHP_FUNCTION(in_array)                                         
{                                                              
    php_search_array(INTERNAL_FUNCTION_PARAM_PASSTHRU, 0);     
}                                                              
/* }}} */                                                      
                                                               
/* {{{ proto mixed array_search(mixed needle, array haystack [, bool strict])
   Searches the array for a given value and returns the corresponding key if successful */
PHP_FUNCTION(array_search)                                     
{                                                              
    php_search_array(INTERNAL_FUNCTION_PARAM_PASSTHRU, 1);     
}                                                              
/* }}} */  

顺便看到了array_search,原来和in_array的内部实现基本一致

其中函数的参数 在./zend.h中

#define INTERNAL_FUNCTION_PARAM_PASSTHRU ht, return_value, return_value_ptr, this_ptr, return_value_used TSRMLS_CC

第二步 在ext/standard/array.c 文件中 查看php_search_array原型

/* void php_search_array(INTERNAL_FUNCTION_PARAMETERS, int behavior)
 * 0 = return boolean
 * 1 = return key
 */      
static void php_search_array(INTERNAL_FUNCTION_PARAMETERS, int behavior) /* {{{ */
{        
    zval *value,            /* value to check for */
         *array,            /* array to check in */
         **entry,           /* pointer to array entry */
         res;              /* comparison result */
    HashPosition pos;        /* hash iterator */
    zend_bool strict = 0;     /* strict comparison or not */
    ulong num_key;
    uint str_key_len;
    char *string_key;
    int (*is_equal_func)(zval *, zval *, zval * TSRMLS_DC) = is_equal_function;
         
    if (zend_parse_parameters(ZEND_NUM_ARGS() TSRMLS_CC, "za|b", &value, &array, &strict) == FAILURE) {
        return;
    }    
         
    if (strict) {
        is_equal_func = is_identical_function;
    }    
         
    zend_hash_internal_pointer_reset_ex(Z_ARRVAL_P(array), &pos);
    while (zend_hash_get_current_data_ex(Z_ARRVAL_P(array), (void **)&entry, &pos) == SUCCESS) {
        is_equal_func(&res, value, *entry TSRMLS_CC);
        if (Z_LVAL(res)) {
            if (behavior == 0) {
                RETURN_TRUE;
            } else {
                /* Return current key */
                switch (zend_hash_get_current_key_ex(Z_ARRVAL_P(array), &string_key, &str_key_len, &num_key, 0, &pos)) {
                    case HASH_KEY_IS_STRING:
                        RETURN_STRINGL(string_key, str_key_len - 1, 1);
                        break;
                    case HASH_KEY_IS_LONG:
                        RETURN_LONG(num_key);
                        break;
                }
            }
        }
        zend_hash_move_forward_ex(Z_ARRVAL_P(array), &pos);
    }    
         
    RETURN_FALSE;
}        
/* }}} */
         
/* {{{ proto bool in_array(mixed needle, array haystack [, bool strict])
   Checks if the given value exists in the array */

我们发现 strict  这个值的不同有两种比较方式,看一下两个函数的不同之处

is_identical_function 检查类型是否相同

ZEND_API int is_identical_function(zval *result, zval *op1, zval *op2 TSRMLS_DC) /* {{{ */
{        
   Z_TYPE_P(result) = IS_BOOL;
   if (Z_TYPE_P(op1) != Z_TYPE_P(op2)) {
      Z_LVAL_P(result) = 0;
      return SUCCESS;
   }    
   switch (Z_TYPE_P(op1)) {
      case IS_NULL:
         Z_LVAL_P(result) = 1;
         break;
      case IS_BOOL:
      case IS_LONG:
      case IS_RESOURCE:
         Z_LVAL_P(result) = (Z_LVAL_P(op1) == Z_LVAL_P(op2));
         break;
      case IS_DOUBLE:
         Z_LVAL_P(result) = (Z_DVAL_P(op1) == Z_DVAL_P(op2));
         break;
      case IS_STRING:
         Z_LVAL_P(result) = ((Z_STRLEN_P(op1) == Z_STRLEN_P(op2))
            && (!memcmp(Z_STRVAL_P(op1), Z_STRVAL_P(op2), Z_STRLEN_P(op1))));
         break;
      case IS_ARRAY:
         Z_LVAL_P(result) = (Z_ARRVAL_P(op1) == Z_ARRVAL_P(op2) 
            zend_hash_compare(Z_ARRVAL_P(op1), Z_ARRVAL_P(op2), (compare_func_t) hash_zval_identical_function, 1 TSRMLS_CC)==0);
         break;
      case IS_OBJECT:
         if (Z_OBJ_HT_P(op1) == Z_OBJ_HT_P(op2)) {
            Z_LVAL_P(result) = (Z_OBJ_HANDLE_P(op1) == Z_OBJ_HANDLE_P(op2));
         } else {
            Z_LVAL_P(result) = 0;
         }
         break;
      default:
         Z_LVAL_P(result) = 0;
         return FAILURE;
   }    
   return SUCCESS;
}        
/* }}} */

 

is_equal_function 不检查类型是否相同,所以需要隐式转换

ZEND_API int is_equal_function(zval *result, zval *op1, zval *op2 TSRMLS_DC) /* {{{ */
{        
   if (compare_function(result, op1, op2 TSRMLS_CC) == FAILURE) {
      return FAILURE;
   }    
   ZVAL_BOOL(result, (Z_LVAL_P(result) == 0));
   return SUCCESS;
}        
/* }}} */

==》compare_function

ZEND_API int compare_function(zval *result, zval *op1, zval *op2 TSRMLS_DC) /* {{{ */
{            
   int ret; 
   int converted = 0;
   zval op1_copy, op2_copy;
   zval *op_free;
             
   while (1) {
      switch (TYPE_PAIR(Z_TYPE_P(op1), Z_TYPE_P(op2))) {
         case TYPE_PAIR(IS_LONG, IS_LONG):
            ZVAL_LONG(result, Z_LVAL_P(op1)>Z_LVAL_P(op2)?1:(Z_LVAL_P(op1)<Z_LVAL_P(op2)?-1:0)); 
            return SUCCESS;
              
         case TYPE_PAIR(IS_DOUBLE, IS_LONG):
            Z_DVAL_P(result) = Z_DVAL_P(op1) - (double)Z_LVAL_P(op2);
            ZVAL_LONG(result, ZEND_NORMALIZE_BOOL(Z_DVAL_P(result)));
            return SUCCESS;
             
         case TYPE_PAIR(IS_LONG, IS_DOUBLE):
            Z_DVAL_P(result) = (double)Z_LVAL_P(op1) - Z_DVAL_P(op2);
            ZVAL_LONG(result, ZEND_NORMALIZE_BOOL(Z_DVAL_P(result)));
            return SUCCESS;
             
         case TYPE_PAIR(IS_DOUBLE, IS_DOUBLE):
            if (Z_DVAL_P(op1) == Z_DVAL_P(op2)) {
               ZVAL_LONG(result, 0);
            } else {
               Z_DVAL_P(result) = Z_DVAL_P(op1) - Z_DVAL_P(op2);
               ZVAL_LONG(result, ZEND_NORMALIZE_BOOL(Z_DVAL_P(result)));
            }
            return SUCCESS;
            
         case TYPE_PAIR(IS_ARRAY, IS_ARRAY):
            zend_compare_arrays(result, op1, op2 TSRMLS_CC);
            return SUCCESS;
            
         case TYPE_PAIR(IS_NULL, IS_NULL):
            ZVAL_LONG(result, 0);
            return SUCCESS;
            
         case TYPE_PAIR(IS_NULL, IS_BOOL):
            ZVAL_LONG(result, Z_LVAL_P(op2) ? -1 : 0);
            return SUCCESS;
             
         case TYPE_PAIR(IS_BOOL, IS_NULL):
            ZVAL_LONG(result, Z_LVAL_P(op1) ? 1 : 0);
            return SUCCESS;
             
         case TYPE_PAIR(IS_BOOL, IS_BOOL):
            ZVAL_LONG(result, ZEND_NORMALIZE_BOOL(Z_LVAL_P(op1) - Z_LVAL_P(op2)));
            return SUCCESS;
             
         case TYPE_PAIR(IS_STRING, IS_STRING):
            zendi_smart_strcmp(result, op1, op2);
            return SUCCESS;
             
         case TYPE_PAIR(IS_NULL, IS_STRING):
            ZVAL_LONG(result, zend_binary_strcmp("", 0, Z_STRVAL_P(op2), Z_STRLEN_P(op2)));
            return SUCCESS;
             
         case TYPE_PAIR(IS_STRING, IS_NULL):
            ZVAL_LONG(result, zend_binary_strcmp(Z_STRVAL_P(op1), Z_STRLEN_P(op1), "", 0));
            return SUCCESS;
             
         case TYPE_PAIR(IS_OBJECT, IS_NULL):
            ZVAL_LONG(result, 1);
            return SUCCESS;
             
         case TYPE_PAIR(IS_NULL, IS_OBJECT):
            ZVAL_LONG(result, -1);
            return SUCCESS;
             
         case TYPE_PAIR(IS_OBJECT, IS_OBJECT):
            /* If both are objects sharing the same comparision handler then use is */
            if (Z_OBJ_HANDLER_P(op1,compare_objects) == Z_OBJ_HANDLER_P(op2,compare_objects)) {
               if (Z_OBJ_HANDLE_P(op1) == Z_OBJ_HANDLE_P(op2)) {
                  /* object handles are identical, apparently this is the same object */
                  ZVAL_LONG(result, 0);
                  return SUCCESS;
               }
               ZVAL_LONG(result, Z_OBJ_HT_P(op1)->compare_objects(op1, op2 TSRMLS_CC));
               return SUCCESS;
            }
            /* break missing intentionally */
             
         default:
            if (Z_TYPE_P(op1) == IS_OBJECT) {
               if (Z_OBJ_HT_P(op1)->get) {
                  op_free = Z_OBJ_HT_P(op1)->get(op1 TSRMLS_CC);
                  ret = compare_function(result, op_free, op2 TSRMLS_CC);
                  zend_free_obj_get_result(op_free TSRMLS_CC);
                  return ret;
               } else if (Z_TYPE_P(op2) != IS_OBJECT && Z_OBJ_HT_P(op1)->cast_object) {
                  ALLOC_INIT_ZVAL(op_free);
                  if (Z_OBJ_HT_P(op1)->cast_object(op1, op_free, Z_TYPE_P(op2) TSRMLS_CC) == FAILURE) {
                     ZVAL_LONG(result, 1);
                     zend_free_obj_get_result(op_free TSRMLS_CC);
                     return SUCCESS;
                  }
                  ret = compare_function(result, op_free, op2 TSRMLS_CC);
                  zend_free_obj_get_result(op_free TSRMLS_CC);
                  return ret;
               }
            }
            if (Z_TYPE_P(op2) == IS_OBJECT) {
               if (Z_OBJ_HT_P(op2)->get) {
                  op_free = Z_OBJ_HT_P(op2)->get(op2 TSRMLS_CC);
                  ret = compare_function(result, op1, op_free TSRMLS_CC);
                  zend_free_obj_get_result(op_free TSRMLS_CC);
                  return ret;
               } else if (Z_TYPE_P(op1) != IS_OBJECT && Z_OBJ_HT_P(op2)->cast_object) {
                  ALLOC_INIT_ZVAL(op_free);
                  if (Z_OBJ_HT_P(op2)->cast_object(op2, op_free, Z_TYPE_P(op1) TSRMLS_CC) == FAILURE) {
                     ZVAL_LONG(result, -1);
                     zend_free_obj_get_result(op_free TSRMLS_CC);
                     return SUCCESS;
                  }
                  ret = compare_function(result, op1, op_free TSRMLS_CC);
                  zend_free_obj_get_result(op_free TSRMLS_CC);
                  return ret;
               } else if (Z_TYPE_P(op1) == IS_OBJECT) {
                  ZVAL_LONG(result, 1);
                  return SUCCESS;
               }
            }
            if (!converted) {
               if (Z_TYPE_P(op1) == IS_NULL) {
                  zendi_convert_to_boolean(op2, op2_copy, result);
                  ZVAL_LONG(result, Z_LVAL_P(op2) ? -1 : 0);
                  return SUCCESS;
               } else if (Z_TYPE_P(op2) == IS_NULL) {
                  zendi_convert_to_boolean(op1, op1_copy, result);
                  ZVAL_LONG(result, Z_LVAL_P(op1) ? 1 : 0);
                  return SUCCESS;
               } else if (Z_TYPE_P(op1) == IS_BOOL) {
                  zendi_convert_to_boolean(op2, op2_copy, result);
                  ZVAL_LONG(result, ZEND_NORMALIZE_BOOL(Z_LVAL_P(op1) - Z_LVAL_P(op2)));
                  return SUCCESS;
               } else if (Z_TYPE_P(op2) == IS_BOOL) {
                  zendi_convert_to_boolean(op1, op1_copy, result);
                  ZVAL_LONG(result, ZEND_NORMALIZE_BOOL(Z_LVAL_P(op1) - Z_LVAL_P(op2)));
                  return SUCCESS;
               } else {
                  zendi_convert_scalar_to_number(op1, op1_copy, result);
                  zendi_convert_scalar_to_number(op2, op2_copy, result);
                  converted = 1;
               }
            } else if (Z_TYPE_P(op1)==IS_ARRAY) {
               ZVAL_LONG(result, 1);
               return SUCCESS;
            } else if (Z_TYPE_P(op2)==IS_ARRAY) {
               ZVAL_LONG(result, -1);
               return SUCCESS;
            } else if (Z_TYPE_P(op1)==IS_OBJECT) {
               ZVAL_LONG(result, 1);
               return SUCCESS;
            } else if (Z_TYPE_P(op2)==IS_OBJECT) {
               ZVAL_LONG(result, -1);
               return SUCCESS;
            } else {
               ZVAL_LONG(result, 0);
               return FAILURE;
            }
      }     
   }         
}             
/* }}} */ 

 接着看一下array 和string 怎么比较的  

==》zend_hash_compare  在Zend/zend_hash.c中

ZEND_API int zend_hash_compare(HashTable *ht1, HashTable *ht2, compare_func_t compar, zend_bool ordered TSRMLS_DC)
{            
     Bucket *p1, *p2 = NULL;
     int result;
     void *pData2;
             
     IS_CONSISTENT(ht1);
     IS_CONSISTENT(ht2);
             
     HASH_PROTECT_RECURSION(ht1);.
     HASH_PROTECT_RECURSION(ht2);.
             
     result = ht1->nNumOfElements - ht2->nNumOfElements;
     if (result!=0) {
          HASH_UNPROTECT_RECURSION(ht1);.
          HASH_UNPROTECT_RECURSION(ht2);.
          return result;
     }        
             
     p1 = ht1->pListHead;
     if (ordered) {
          p2 = ht2->pListHead;
     }         
             
     while (p1) {
          if (ordered && !p2) {
               HASH_UNPROTECT_RECURSION(ht1);.
               HASH_UNPROTECT_RECURSION(ht2);.
               return 1; /* That's not supposed to happen */
          }    
          if (ordered) {
               if (p1->nKeyLength==0 && p2->nKeyLength==0) { /* numeric indices */
                    result = p1->h - p2->h;
                    if (result!=0) {
                         HASH_UNPROTECT_RECURSION(ht1);.
                         HASH_UNPROTECT_RECURSION(ht2);.
                         return result;
                    }
               } else { /* string indices */
                    result = p1->nKeyLength - p2->nKeyLength;
                    if (result!=0) {
                         HASH_UNPROTECT_RECURSION(ht1);.
                         HASH_UNPROTECT_RECURSION(ht2);.
                         return result;
                    }
                    result = memcmp(p1->arKey, p2->arKey, p1->nKeyLength);
                    if (result!=0) {
                         HASH_UNPROTECT_RECURSION(ht1);.
                         HASH_UNPROTECT_RECURSION(ht2);.
                         return result;
                    }
               }
               pData2 = p2->pData;
          } else {
               if (p1->nKeyLength==0) { /* numeric index */
                    if (zend_hash_index_find(ht2, p1->h, &pData2)==FAILURE) {
                         HASH_UNPROTECT_RECURSION(ht1);.
                         HASH_UNPROTECT_RECURSION(ht2);.
                         return 1;
                    }
               } else { /* string index */
                    if (zend_hash_quick_find(ht2, p1->arKey, p1->nKeyLength, p1->h, &pData2)==FAILURE) { 
                         HASH_UNPROTECT_RECURSION(ht1);.
                         HASH_UNPROTECT_RECURSION(ht2);.
                         return 1;
                    }
               } 
          }     
          result = compar(p1->pData, pData2 TSRMLS_CC);
          if (result!=0) {
               HASH_UNPROTECT_RECURSION(ht1);.
               HASH_UNPROTECT_RECURSION(ht2);.
               return result;
          }     
          p1 = p1->pListNext;
          if (ordered) {
               p2 = p2->pListNext;
          }     
     }         
               
     HASH_UNPROTECT_RECURSION(ht1);.
     HASH_UNPROTECT_RECURSION(ht2);.
     return 0; 
}

==》还有一个 zendi_smart_strcmp  在Zend/zend_operators.c中

ZEND_API void zendi_smart_strcmp(zval *result, zval *s1, zval *s2) /* {{{ */
{    
     int ret1, ret2;
     int oflow1, oflow2;
     long lval1, lval2;
     double dval1, dval2;
     
     if ((ret1=is_numeric_string_ex(Z_STRVAL_P(s1), Z_STRLEN_P(s1), &lval1, &dval1, 0, &oflow1)) &&
          (ret2=is_numeric_string_ex(Z_STRVAL_P(s2), Z_STRLEN_P(s2), &lval2, &dval2, 0, &oflow2))) {
#if ULONG_MAX == 0xFFFFFFFF
          if (oflow1 != 0 && oflow1 == oflow2 && dval1 - dval2 == 0. &&
               ((oflow1 == 1 && dval1 > 9007199254740991. /*0x1FFFFFFFFFFFFF*/)
                    (oflow1 == -1 && dval1 < -9007199254740991.))) {
#else
          if (oflow1 != 0 && oflow1 == oflow2 && dval1 - dval2 == 0.) {
#endif
               /* both values are integers overflown to the same side, and the
                * double comparison may have resulted in crucial accuracy lost */
               goto string_cmp;
          }
          if ((ret1==IS_DOUBLE)      (ret2==IS_DOUBLE)) {
               if (ret1!=IS_DOUBLE) {
                    if (oflow2) {
                         /* 2nd operand is integer > LONG_MAX (oflow2==1) or < LONG_MIN (-1) */
                         ZVAL_LONG(result, -1 * oflow2);
                         return;
                    }
                    dval1 = (double) lval1;
               } else if (ret2!=IS_DOUBLE) {
                    if (oflow1) {
                         ZVAL_LONG(result, oflow1);
                         return;
                    }
                    dval2 = (double) lval2;
               } else if (dval1 == dval2 && !zend_finite(dval1)) {
                    /* Both values overflowed and have the same sign,
                     * so a numeric comparison would be inaccurate */
                    goto string_cmp;
               }
               Z_DVAL_P(result) = dval1 - dval2;
               ZVAL_LONG(result, ZEND_NORMALIZE_BOOL(Z_DVAL_P(result)));
          } else { /* they both have to be long's */
               ZVAL_LONG(result, lval1 > lval2 ? 1 : (lval1 < lval2 ? -1 : 0));
          }
     } else {
string_cmp:
          Z_LVAL_P(result) = zend_binary_zval_strcmp(s1, s2);
          ZVAL_LONG(result, ZEND_NORMALIZE_BOOL(Z_LVAL_P(result)));
     }
}   
/* }}} *

我们这个类型转换函数  is_numeric_string_ex  在 Zend/zend_operators.h

继续跟踪

==》is_numeric_string_ex

static inline zend_uchar is_numeric_string_ex(const char *str, int length, long *lval, double *dval, int allow_errors, int *oflow_info)
{
     const char *ptr;
     int base = 10, digits = 0, dp_or_e = 0;
     double local_dval;
     zend_uchar type;
 
     if (!length) {
          return 0;
     }
 
     if (oflow_info != NULL) {
          *oflow_info = 0;
     }
 
     /* Skip any whitespace
      * This is much faster than the isspace() function */
     while (*str == ' '      *str == '\t'      *str == '\n'      *str == '\r'      *str == '\v'      *str == '\f') {
          str++;
          length--;
     }
     ptr = str;
 
     if (*ptr == '-'      *ptr == '+') {
          ptr++;
     }
 
     if (ZEND_IS_DIGIT(*ptr)) {
          /* Handle hex numbers
           * str is used instead of ptr to disallow signs and keep old behavior */
          if (length > 2 && *str == '0' && (str[1] == 'x'      str[1] == 'X')) {
               base = 16;
               ptr += 2;
          }
 
          /* Skip any leading 0s */
          while (*ptr == '0') {
               ptr++;
          }
 
          /* Count the number of digits. If a decimal point/exponent is found,
           * it's a double. Otherwise, if there's a dval or no need to check for
           * a full match, stop when there are too many digits for a long */
          for (type = IS_LONG; !(digits >= MAX_LENGTH_OF_LONG && (dval      allow_errors == 1)); digits++, ptr++) {
check_digits:
               if (ZEND_IS_DIGIT(*ptr)      (base == 16 && ZEND_IS_XDIGIT(*ptr))) {
                    continue;
               } else if (base == 10) {
                    if (*ptr == '.' && dp_or_e < 1) {
                         goto process_double;
                    } else if ((*ptr == 'e'      *ptr == 'E') && dp_or_e < 2) {
                         const char *e = ptr + 1;
 
                         if (*e == '-'      *e == '+') {
                              ptr = e++;
                         }
                         if (ZEND_IS_DIGIT(*e)) {
                              goto process_double;
                         }
                    }
               }
 
               break;
          }
 
          if (base == 10) {
               if (digits >= MAX_LENGTH_OF_LONG) {
                    if (oflow_info != NULL) {
                         *oflow_info = *str == '-' ? -1 : 1;
                    }
                    dp_or_e = -1;
                    goto process_double;
               }
          } else if (!(digits < SIZEOF_LONG * 2      (digits == SIZEOF_LONG * 2 && ptr[-digits] <= '7'))) {
               if (dval) {
                    local_dval = zend_hex_strtod(str, &ptr);
               }
               if (oflow_info != NULL) {
                    *oflow_info = 1;
               }
               type = IS_DOUBLE;
          }
     } else if (*ptr == '.' && ZEND_IS_DIGIT(ptr[1])) {
process_double:
          type = IS_DOUBLE;
 
          /* If there's a dval, do the conversion; else continue checking
           * the digits if we need to check for a full match */
          if (dval) {
               local_dval = zend_strtod(str, &ptr);
          } else if (allow_errors != 1 && dp_or_e != -1) {
               dp_or_e = (*ptr++ == '.') ? 1 : 2;
               goto check_digits;
          }
     } else {
          return 0;
     }
 
     if (ptr != str + length) {
          if (!allow_errors) {
               return 0;
          }
          if (allow_errors == -1) {
               zend_error(E_NOTICE, "A non well formed numeric value encountered");
          }
     }
 
     if (type == IS_LONG) {
          if (digits == MAX_LENGTH_OF_LONG - 1) {
               int cmp = strcmp(&ptr[-digits], long_min_digits);
 
               if (!(cmp < 0      (cmp == 0 && *str == '-'))) {
                    if (dval) {
                         *dval = zend_strtod(str, NULL);
                    }
                    if (oflow_info != NULL) {
                         *oflow_info = *str == '-' ? -1 : 1;
                    }
 
                    return IS_DOUBLE;
               }
          }
 
          if (lval) {
               *lval = strtol(str, NULL, base);
          }
 
          return IS_LONG;
     } else {
          if (dval) {
               *dval = local_dval;
          }
 
          return IS_DOUBLE;
     }
}

type == IS_LONG时果然有这个代码

截取这一段代码

……
if (type == IS_LONG) {
     if (digits == MAX_LENGTH_OF_LONG - 1) {
          int cmp = strcmp(&ptr[-digits], long_min_digits);

          if (!(cmp < 0      (cmp == 0 && *str == '-'))) {
               if (dval) {
                    *dval = zend_strtod(str, NULL);
               }
               if (oflow_info != NULL) {
                    *oflow_info = *str == '-' ? -1 : 1;
               }

               return IS_DOUBLE;
          }
     }

     if (lval) {
          *lval = strtol(str, NULL, base);
     }

     return IS_LONG;
}
……

原来如此  strtol 在这里的

www.bkjia.comtruehttp://www.bkjia.com/PHPjc/1069125.htmlTechArticle深入PHP内核之in_array,php内核in_array 无意中看到一段代码 1、a.php ?php$y="12";$x = array();for($j=0;$j50000;$j++){ $x[]= "{$j}";}for($i=0;$i30000;$i++){ if(in_ar...

相关文章

PHP速学教程(入门到精通)
PHP速学教程(入门到精通)

PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

16

2026.03.11

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

23

2026.03.10

Kotlin Android模块化架构与组件化开发实践
Kotlin Android模块化架构与组件化开发实践

本专题围绕 Kotlin 在 Android 应用开发中的架构实践展开,重点讲解模块化设计与组件化开发的实现思路。内容包括项目模块拆分策略、公共组件封装、依赖管理优化、路由通信机制以及大型项目的工程化管理方法。通过真实项目案例分析,帮助开发者构建结构清晰、易扩展且维护成本低的 Android 应用架构体系,提升团队协作效率与项目迭代速度。

75

2026.03.09

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

95

2026.03.06

Rust内存安全机制与所有权模型深度实践
Rust内存安全机制与所有权模型深度实践

本专题围绕 Rust 语言核心特性展开,深入讲解所有权机制、借用规则、生命周期管理以及智能指针等关键概念。通过系统级开发案例,分析内存安全保障原理与零成本抽象优势,并结合并发场景讲解 Send 与 Sync 特性实现机制。帮助开发者真正理解 Rust 的设计哲学,掌握在高性能与安全性并重场景中的工程实践能力。

218

2026.03.05

PHP高性能API设计与Laravel服务架构实践
PHP高性能API设计与Laravel服务架构实践

本专题围绕 PHP 在现代 Web 后端开发中的高性能实践展开,重点讲解基于 Laravel 框架构建可扩展 API 服务的核心方法。内容涵盖路由与中间件机制、服务容器与依赖注入、接口版本管理、缓存策略设计以及队列异步处理方案。同时结合高并发场景,深入分析性能瓶颈定位与优化思路,帮助开发者构建稳定、高效、易维护的 PHP 后端服务体系。

420

2026.03.04

AI安装教程大全
AI安装教程大全

2026最全AI工具安装教程专题:包含各版本AI绘图、AI视频、智能办公软件的本地化部署手册。全篇零基础友好,附带最新模型下载地址、一键安装脚本及常见报错修复方案。每日更新,收藏这一篇就够了,让AI安装不再报错!

168

2026.03.04

Swift iOS架构设计与MVVM模式实战
Swift iOS架构设计与MVVM模式实战

本专题聚焦 Swift 在 iOS 应用架构设计中的实践,系统讲解 MVVM 模式的核心思想、数据绑定机制、模块拆分策略以及组件化开发方法。内容涵盖网络层封装、状态管理、依赖注入与性能优化技巧。通过完整项目案例,帮助开发者构建结构清晰、可维护性强的 iOS 应用架构体系。

222

2026.03.03

C++高性能网络编程与Reactor模型实践
C++高性能网络编程与Reactor模型实践

本专题围绕 C++ 在高性能网络服务开发中的应用展开,深入讲解 Socket 编程、多路复用机制、Reactor 模型设计原理以及线程池协作策略。内容涵盖 epoll 实现机制、内存管理优化、连接管理策略与高并发场景下的性能调优方法。通过构建高并发网络服务器实战案例,帮助开发者掌握 C++ 在底层系统与网络通信领域的核心技术。

33

2026.03.03

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PHP基础入门课程
PHP基础入门课程

共33课时 | 2.2万人学习

Go语言教程手册
Go语言教程手册

共23课时 | 18万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号