我正在用纯PHP创建一个Web应用程序,并且我必须在函数之间交换某些状态,该状态大约可以是5,例如:user_created,user_active,user_inactive等。那么,最有效的方法是什么?
最初我是从函数中返回字符串,但是我知道字符串比较可能会很慢,而且如果状态名称越来越长,则更多,所以我考虑创建带有整数值的常量,但这让我受益匪浅思考...如果PHP解析器在比较之前仍然需要读取整个常量名称,这与使用字符串或变量不同吗?有更有效的方法吗?
对不起,如果它看起来像一个琐碎的问题,我只是尝试优化应用程序的各个方面
如果不清楚我要做什么,那真的很简单:
function getUserStatus(){
//return some value
}
然后按如下所示使用它
$userStatus = getUserStatus
switch ($userStatus){
// all posible cases
}
答案 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