>  기사  >  백엔드 개발  >  PHP의 strpos 함수에 대한 심층 분석

PHP의 strpos 함수에 대한 심층 분석

不言
不言원래의
2018-07-28 10:10:012129검색

이 기사의 내용은 PHP의 strpos 기능에 대한 심층 분석을 공유하는 것입니다. 내용이 매우 상세하므로 도움이 필요한 친구들이 참고할 수 있기를 바랍니다.

개요

PHP에서 strpos는 문자열이 다른 문자열에 존재하는지 확인하는 데 자주 사용됩니다. 이 문서에서는 strpos 함수와 그 구현을 소개합니다. strpos 判断字符串是否在另一个字符串中存在, 本文介绍 strpos 函数及其实现。

strpos应用

<?php
/* strpos示例 */

// test
echo &#39;match:&#39;, strpos(&#39;xasfsdfbk&#39;, &#39;xasfsdfbk&#39;) !== false ? &#39;true&#39; : &#39;false&#39;, &#39;;&#39;, PHP_EOL;
echo &#39;match:&#39;, strpos(&#39;xasfsdfbk&#39;, &#39;fbk&#39;) !== false ? &#39;true&#39; : &#39;false&#39;, &#39;;&#39;, PHP_EOL;
echo &#39;match:&#39;, strpos(&#39;xasfsdfbk&#39;, &#39;xs&#39;) != false ? &#39;true&#39; : &#39;false&#39;, &#39;;&#39;, PHP_EOL;
echo &#39;match:&#39;, strpos(&#39;xasfsdfbk&#39;, &#39;sfs&#39;) !== false ? &#39;true&#39; : &#39;false&#39;, &#39;;&#39;, PHP_EOL;

// code
strpos(&#39;xasfsdfbk&#39;, &#39;sfs&#39;);
Warning: strpos 函数可能返回布尔值 FALSE,但也可能返回等同于 FALSE 的非布尔值。请阅读 布尔类型章节以获取更多信息。应使用 ===
strpos application

PHP_FUNCTION(strpos)
{
    zval *needle;
    zend_string *haystack;
    char *found = NULL;
    char  needle_char[2];
    zend_long  offset = 0;

#ifndef FAST_ZPP
    if (zend_parse_parameters(ZEND_NUM_ARGS(), "Sz|l", &haystack, &needle, &offset) == FAILURE) {
        return;
    }
#else
    ZEND_PARSE_PARAMETERS_START(2, 3)
        Z_PARAM_STR(haystack)
        Z_PARAM_ZVAL(needle)
        Z_PARAM_OPTIONAL
        Z_PARAM_LONG(offset)
    ZEND_PARSE_PARAMETERS_END();
#endif

    if (offset < 0) {
        offset += (zend_long)ZSTR_LEN(haystack);
    }
    if (offset < 0 || (size_t)offset > ZSTR_LEN(haystack)) {
        php_error_docref(NULL, E_WARNING, "Offset not contained in string");
        RETURN_FALSE;
    }

    if (Z_TYPE_P(needle) == IS_STRING) {
        if (!Z_STRLEN_P(needle)) {
            php_error_docref(NULL, E_WARNING, "Empty needle");
            RETURN_FALSE;
        }

        found = (char*)php_memnstr(ZSTR_VAL(haystack) + offset,
                            Z_STRVAL_P(needle),
                            Z_STRLEN_P(needle),
                            ZSTR_VAL(haystack) + ZSTR_LEN(haystack));
    } else {
        if (php_needle_char(needle, needle_char) != SUCCESS) {
            RETURN_FALSE;
        }
        needle_char[1] = 0;

        found = (char*)php_memnstr(ZSTR_VAL(haystack) + offset,
                            needle_char,
                            1,
                            ZSTR_VAL(haystack) + ZSTR_LEN(haystack));
    }

    if (found) {
        RETURN_LONG(found - ZSTR_VAL(haystack));
    } else {
        RETURN_FALSE;
    }
}
경고: strpos 이 함수는 부울 값 FALSE를 반환할 수 있지만 에 해당하는 값을 반환할 수도 있습니다. FALSE code>에 대한 부울이 아닌 값입니다. 자세한 내용은 부울 유형 장을 읽어보세요. 이 함수의 반환 값은 === 연산자를 사용하여 테스트해야 합니다. strpos 계열 함수FunctionDescriptionVersionstrpos문자열의 첫 번째 항목 찾기PHP 4, PHP 5, PHP 7스트라이포스 문자열이 처음 나타나는 위치 찾기(대소문자 구분 안 함) PHP 5, PHP 7strrpos대상 문자열에서 지정된 문자열이 마지막으로 나타나는 위치 계산PHP 4, PHP 5, PHP 7strripos대상 문자열에서 지정된 문자열이 마지막으로 나타나는 위치 계산(대소문자 구분 안 함)PHP 5, PHP 7mb_strpos다른 문자열에서 문자열 찾기 찾기 PHP 4 >= 4.0.6, PHP 5, PHP 7strstr에서 처음으로 나타나는 문자열 찾기 PHP 4, PHP 5, PHP 7strstrStrstr() 함수 대소문자 무시 버전PHP 4, PHP 5, PHP 7substr_count문자열 발생 횟수를 계산합니다PHP 4, PHP 5, PHP 7

mb* 관련 함수도 사용할 수 있습니다. 예를 들어 mb_strpos는 문자 수에 따라 멀티바이트 안전 strpos() 작업을 수행합니다.

PHP(strpos) 소스 코드
  • strpos(ext/standard/string.c)

PHP 소스 코드 주소

    #define php_memnstr zend_memnstr /* 338 line*/
  • php_memnstr(main/php.h)

  • PHP 소스 코드 주소

    /*
     * 此函数的作用是在haystack中查找needle,如果不存在返回null,如果存在,返回指向haystack中needle头字符的指针
     */
    zend_memnstr(const char *haystack, const char *needle, size_t needle_len, const char *end)
    {
        const char *p = haystack;
        const char ne = needle[needle_len-1];
        ptrdiff_t off_p;
        size_t off_s;
    
        if (needle_len == 1) {
            return (const char *)memchr(p, *needle, (end-p));
        }
    
        off_p = end - haystack;
        off_s = (off_p > 0) ? (size_t)off_p : 0;
    
        if (needle_len > off_s) {
            return NULL;
        }
    
        if (EXPECTED(off_s < 1024 || needle_len < 3)) {
            // 第一个优化,只查找end - needle_len次
            end -= needle_len;
    
            while (p <= end) {
                // 第二个优化,先判断字符串的开头和结尾是否一样再判断整个字符串
                if ((p = (const char *)memchr(p, *needle, (end-p+1))) && ne == p[needle_len-1]) {
                    if (!memcmp(needle, p, needle_len-1)) {
                        return p;
                    }
                }
    
                if (p == NULL) {
                    return NULL;
                }
    
                p++;
            }
    
            return NULL;
        } else {
            return zend_memnstr_ex(haystack, needle, needle_len, end);
        }
    }
  • zend_memnstr(Zend/zend_operators.h)

  • PHP 소스 코드 주소

      /*
      头文件:#include <string.h>
      
      定义函数:void * memchr(const void *s, char c, size_t n);
      
      函数说明:memchr()从头开始搜寻s 所指的内存内容前n 个字节,直到发现第一个值为c 的字节,则返回指向该字节的指针。
      
      返回值:如果找到指定的字节则返回该字节的指针,否则返回0。
      */
      
      #ifndef __HAVE_ARCH_MEMCHR
      void *memchr(const void *s, int c, size_t n)
      {
          const unsigned char *p = s;
          while (n-- != 0) {
                  if ((unsigned char)c == *p++) {
                  return (void *)(p - 1);
              }
          }
          return NULL;
      }
      EXPORT_SYMBOL(memchr);
      #endif
    • memchr(string.h)

    • Linux 커널 버전 - 소스 코드 주소

      /* 字符串函数memcmp
         原型:extern int memcmp(void *buf1, void *buf2, unsigned int count); 
         功能:比较内存区域buf1和buf2的前count个字节
         说明:当buf1<buf2时,返回值<0  
               当buf1=buf2时,返回值=0   
               当buf1>buf2时,返回值>0                                        
      */
      #ifndef __HAVE_ARCH_MEMCMP
      #undef memcmp
      __visible int memcmp(const void *cs, const void *ct, size_t count)
      {
          const unsigned char *su1, *su2;
          int res = 0;
      
          for (su1 = cs, su2 = ct; 0 < count; ++su1, ++su2, count--)
              if ((res = *su1 - *su2) != 0)
                  break;
          return res;
      }
      EXPORT_SYMBOL(memcmp);
      #endif
    • memcmp(문자열 .h)

    • Linux 커널 버전 - 소스 코드 주소

      rrreee

      Tips

      strpos 함수는 대소문자를 구분합니다.

      관련 권장 사항:

      PHP에서 잠금 메커니즘 적용
      PHP ID를 암호화하는 방법🎜🎜🎜

      위 내용은 PHP의 strpos 함수에 대한 심층 분석의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

    성명:
    본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.