php strpose_深入理解PHP之strpos

概述

在php中经常用 strpos 判断字符串是否在另一个字符串中存在, 本文介绍 strpos 函数及其实现。

strpos应用

<?php

/* strpos示例 */

// test

echo 'match:', strpos('xasfsdfbk', 'xasfsdfbk') !== false ? 'true' : 'false', ';', PHP_EOL;

echo 'match:', strpos('xasfsdfbk', 'fbk') !== false ? 'true' : 'false', ';', PHP_EOL;

echo 'match:', strpos('xasfsdfbk', 'xs') != false ? 'true' : 'false', ';', PHP_EOL;

echo 'match:', strpos('xasfsdfbk', 'sfs') !== false ? 'true' : 'false', ';', PHP_EOL;

// code

strpos('xasfsdfbk', 'sfs');

Warning: strpos 函数可能返回布尔值 FALSE,但也可能返回等同于 FALSE 的非布尔值。请阅读 布尔类型章节以获取更多信息。应使用 === 运算符来测试此函数的返回值。

strpos系列函数

函数

描述

版本

查找字符串首次出现的位置

PHP 4, PHP 5, PHP 7

查找字符串首次出现的位置(不区分大小写)

PHP 5, PHP 7

计算指定字符串在目标字符串中最后一次出现的位置

PHP 4, PHP 5, PHP 7

计算指定字符串在目标字符串中最后一次出现的位置(不区分大小写)

PHP 5, PHP 7

查找字符串在另一个字符串中首次出现的位置

PHP 4 >= 4.0.6, PHP 5, PHP 7

查找字符串的首次出现

PHP 4, PHP 5, PHP 7

strstr() 函数的忽略大小写版本

PHP 4, PHP 5, PHP 7

计算字串出现的次数

PHP 4, PHP 5, PHP 7

mb* 相关的函数也可, 比如说mb_strpos是基于字符数执行一个多字节安全的 strpos() 操作。

PHP(strpos)源码

strpos(ext/standard/string.c)

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;

}

}

php_memnstr(main/php.h)

define php_memnstr zend_memnstr /* 338 line*/

zend_memnstr(Zend/zend_operators.h)

/*

此函数的作用是在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);

}

}

memchr(string.h)

/*

头文件:#include

定义函数: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

memcmp(string.h)

/* 字符串函数memcmp

原型:extern int memcmp(void *buf1, void *buf2, unsigned int count);

功能:比较内存区域buf1和buf2的前count个字节

说明:当buf1

当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

提示

strpos函数对大小写敏感。

参考

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值