常见的php函数代码性能对比

Posted by fsoooo Blog on June 11, 2022

日常开发中,我们通常会使用常规的、显而易见的 PHP 函数来解决相应的问题。

但是这些方便的内置函数真的高效吗?有没有更高效的替代解决方案?

现在,来介绍其中的一些替代方案。

如果你正在寻找在生产中进一步减少执行时间的可能性,这将非常有用。

让我们来看看哪些 PHP 方法可能会被性能更好的方法取代,以及是否有任何成本或权衡。

所有这些方法都在本地 Web 服务器上使用 PHP 7.4 进行了测试。

1. 删除重复项

你有一个包含重复项的大型数组,并且希望删除它们,以便只有一个具有唯一值的数组。

 常规

array_unique($array);

 其他方式

array_keys(array_flip($array));

####性能

创建了一个 400 万的数组,其中有 300 万个重复项,这是最好的优化结果:

方法执行时间

方法 执行时间
array_unique 787.31 ms
array_keys array_flip 434.03 ms

其他方式比常规方式 快了 1.8 倍 (44.87%) . 平均也有大约 1.5 倍 (30%) 的提升。

#####总结:这仅适用于一维数组,因为 array_flip 会反转键和值.

2. 获取随机的数组元素

你想从一个大数组中获取一个随机值

常规

array_rand($array);

其他方式

$array[mt_rand(0, count($array) - 1)];

#### 性能

我创建了一个包含 500 万个元素的数组,这是最佳结果:

方法执行时间

方法 执行时间
array_rand 25.99 μs
mt_rand 0.95 μs

替代方法在此测量中快 27.3 倍 (96.33%)。平均快 8 倍 (87%)。这个结果特别令人惊讶,因为 mt_rand 是 Mersenne Twister 随机数生成器的实现,而且从 PHP7.1 开始,内部随机化算法已更改为使用完全相同的算法。

3. 字母数字字符测试

如果你有一个字符串,并且希望测试它是否只包含字母数字字符。

常规

preg_match('/[a-zA-Z0-9]+/', $string);

其他方式

ctype_alnum($string);

####性能

创建了一个包含超过 100k 字母数字和非字母数字字符串的数组。以下是排名靠前的结果:

methodexecution time
preg_match 15.39 ms
ctype_alnum 2.06 ms

替代方法在此测量中快 7.5 倍 (86.59%)。平均而言,它快了约 4 倍 (76%)。

同样的道理也可以应用于 ctype_alpha() 命令 (检查字母字符) 和 ctype_digit() 命令 (检查数字字符)。

4. 替换子字符串

你有一个字符串,并且希望用另一个子字符串替换它的一部分。

 常规

str_replace('a', 'b', $string);

 其他方式

strtr($string, 'a', 'b');

####性能

创建了一个包含 500 万个随机字符串的数组。以下是排名靠前的结果:

methodexecution time
str_replace 676.59 ms
strtr 305.59 ms

替代方法在此测量中快 2.2 倍 (54.83%)。平均快 2 倍 (51%)。

额外的性能改进

以下是我在编码约定中搜集的一些附加方法,我发现它们可以略微提高性能 (如果适用):

  • 更喜欢 JSON 而不是 XML
  • 在之前声明变量,而不是在循环的每次迭代中声明变量
  • 避免循环头部中的函数调用 (在每次迭代中调用的 for ($i=0; $i<count($array); $i) 中的 count() 中)
  • 注销消耗内存变量
  • 首选 select 语句而不是多个 if 语句
  • 优先选择 require/include 而不是 require_once/include_once (确保正确的操作码缓存)

最后一句话:我知道关于过早优化的讨论。我同意,生产中的性能取决于诸如数据库查询之类的瓶颈,在处理性能时应该重点关注这些瓶颈。但我认为,如果有更快的替代方案,例如,在 regex 更容易处理和维护的情况下,为什么不使用它们呢?

总结

我们已经看到,即使使用当前的 PHP 7.4 (已经比以前的 PHP 版本快得多),也有可能使用其他方法进一步提高脚本性能。