PHP使用常量而不是字符串的性能

时间:2018-08-22 04:09:11

标签: php php-7

我正在用纯PHP创建一个Web应用程序,并且我必须在函数之间交换某些状态,该状态大约可以是5,例如:user_created,user_active,user_inactive等。那么,最有效的方法是什么?

最初我是从函数中返回字符串,但是我知道字符串比较可能会很慢,而且如果状态名称越来越长,则更多,所以我考虑创建带有整数值的常量,但这让我受益匪浅思考...如果PHP解析器在比较之前仍然需要读取整个常量名称,这与使用字符串或变量不同吗?有更有效的方法吗?

对不起,如果它看起来像一个琐碎的问题,我只是尝试优化应用程序的各个方面

如果不清楚我要做什么,那真的很简单:

function getUserStatus(){
    //return some value
}

然后按如下所示使用它

$userStatus = getUserStatus
    switch ($userStatus){
    // all posible cases
}

3 个答案:

答案 0 :(得分:2)

  

我知道字符串比较会很慢

这在网络应用程序的上下文中并不是很准确。

字符串比较需要微秒

加载网页,访问数据库等操作需要毫秒 seconds (长数千倍)。

因此,这种微优化几乎是不值得的。相反,您应该关注使代码最清晰,最易于维护的内容。

如果您要遍历数百万个项目或每秒处理数千个请求,则稍后将对代码进行性能优化。

  

如果PHP解析器仍然需要在比较之前读取整个常量名称,这与使用字符串或变量不同吗?

使用常量(并使常量的值为数字数字而不是字符串)可能比使用字符串值要快一些,并且代码也可能更清晰和可维护。但同样,在加载网页的过程中,您将无法分辨出差异。

答案 1 :(得分:2)

我将解决您的问题的多个方面,以期使您有更广泛的了解。

对于您的实际问题,答案是应用程序的运行时阶段和解析阶段是独立的概念。 PHP确实必须解析您的源代码,但是该源代码随后被转换为另一种称为Opcode的格式。操作码是代码的较低级别,更有效的表示形式。运行时还会缓存此表示,这意味着它不必在每次调用时都解析您的源代码。

比较两个字符串时,所花费的时间取决于最短字符串的长度(通常而言)。但是,通常可以更有效地比较整数,这就是您试图通过使用基于整数的常量获得的性能优势。查找常量的值不一定需要将常量的名称作为字符串进行遍历,因为它在操作码中的表示形式不一定与源代码中的表示形式相同。但是,正如在另一个答案中指出的那样,似乎PHP实际上已针对字符串比较进行了优化,并且实际上在很小的程度上胜过了常量整数比较!

但这不一定意味着您无论如何都不应该切换到常量。在这些类型的比较中使用常量vs字符串文字的真正好处是,运行时能够在您打错字时为您提供帮助。如果您的代码库中堆满了类型为$var == "some special value"的字符串比较,那么解析器很乐意让您在代码"some special value"的晦涩之处引入错字。但是,如果您使用常量,则如果您输错常量的名称,则类似$var == MY_SPECIAL_CONSTANT的比较会警告您。它还允许您在单个位置(常量定义)中定义具体的值,这样,如果您需要更改它,则只需一行操作即可完成,而无需进行大量的查找替换工作。

许多PHP代码倾向于“基于字符串的编程”,而这些类型的代码库可能很脆弱,很容易受到拼写错误的影响。常量确实可以在这里及其价值所在的地方有所帮助,而不是在性能方面。

答案 2 :(得分:1)

您可以使用以下类似的方法对其进行测试。

请注意,PHP进行了很多优化,并将已编译的字节码存储在其缓存中。 http://php.net/manual/en/intro.opcache.php

结果

字符串    0.38328790664673    不变    0.50211310386658

字符串    0.38391804695129    不变    0.51568698883057

让我惊讶的是String似乎更快。

我注意到opcache配置中的以下设置:

opcache.interned_strings_buffer integer
The amount of memory used to store interned strings, in megabytes. 
This configuration directive is ignored in PHP < 5.3.0.

一个非常简洁的设置,包含0个文档。 PHP使用一种称为字符串实习的技术来提高性能,例如,如果您在代码中使用字符串“ foobar” 1000次,则PHP在内部将为该字符串存储1个不可变变量,而对于另一个999使用指向它的指针您使用它的次数。凉。此设置将其带入一个新的水平。而不是每个php-fpm进程都有这些不可变字符串的池,此设置在所有php-fpm进程中共享它。它可以节省内存并提高性能,特别是在大型应用程序中。

因此,在PHP中声明字符串比较比常量比较慢是错误的假设。

但是:您可以破坏这个优化示例:

$state = "State";
switch($string) {
    case "Offline" . $state:
    break;
}

其结果将是: 字符串0.61401081085205常数0.51961803436279

在这种情况下,常数比较会更快。

PHP5.4和RFC中增加了性能改进 https://wiki.php.net/rfc/performanceimprovements

但是请注意,常量通常可以更好地重构代码并具有更好的可维护性。此外,性能影响可以忽略不计

function doSomethingString() {
    return "OfflineState";
}

const OFFLINE_STATE = 1;
function doSomethingConstant() {
    return OFFLINE_STATE;
}

function dummy() {}

// String
echo('string' . PHP_EOL);
$start = microtime(true);
for($i = 0; $i < 10000000; $i++) {
    switch(doSomethingString()) {
        case "OfflineState":
            dummy();
            break;
    }
}
echo(PHP_EOL);
$end = microtime(true);
echo($end - $start);
echo(PHP_EOL);


//Constant
echo('constant' . PHP_EOL);
$start = microtime(true);
for($i = 0; $i < 10000000; $i++) {
    switch(doSomethingConstant()) {
        case OFFLINE_STATE:
            dummy();
            break;
    }
}
echo(PHP_EOL);
$end = microtime(true);
echo($end - $start);
echo(PHP_EOL);

我的php版本:

PHP 7.2.8-1+ubuntu18.04.1+deb.sury.org+1 (cli) (built: Jul 25 2018 10:52:19) ( NTS )
Copyright (c) 1997-2018 The PHP Group
Zend Engine v3.2.0, Copyright (c) 1998-2018 Zend Technologies
    with Zend OPcache v7.2.8-1+ubuntu18.04.1+deb.sury.org+1, Copyright (c) 1999-2018, by Zend Technologies