12

我对 PHP 比较陌生,并且正在慢慢学习特定于该语言的特性。我被很多人讨厌的一件事是我(有人告诉我)使用了太多的函数调用,并且通常被要求做一些事情来解决它们。这里有两个例子:

// Change this:
} catch (Exception $e) {
  print "It seems that error " . $e->getCode() . " occured";
  log("Error: " . $e->getCode());
}

// To this:
} catch (Exception $e) {
  $code = $e->getCode();
  print "It seems that error " . $code . " occured";
  log("Error: " . $code);
}

第二个例子

// Change this:
$customer->setProducts($products);

// To this:
if (!empty($products)) {
  $customer->setProducts($products);
}

在第一个示例中,我发现分配$e->getCode()$code广告会产生轻微的认知开销;“'$code'是什么?啊,这是异常的代码。” 而第二个例子增加了圈复杂度。在这两个示例中,我发现优化是以牺牲可读性和可维护性为代价的。

性能提升值得还是这种微优化?

我应该注意到,我们现在坚持使用 PHP 5.2。

我进行了一些非常粗略的基准测试,发现函数调用性能命中率约为 10% 到 70%,具体取决于基准测试的性质。我承认这很重要。但在那个 catch 块被命中之前,有一个对数据库和 HTTP 端点的调用。在$products设置之前,数组$customer发生了复杂的排序。归根结底,这种优化是否证明了使代码更难阅读和维护的成本是合理的?或者,尽管这些示例是简化的,但是否有人发现第二个示例与第一个示例一样容易或更容易阅读(我是维纳)?$products

任何人都可以引用任何关于此的好文章或研究吗?

编辑:

一个示例台架测试:

<?php
class Foo {
        private $list;
        public function setList($list) {
                $this->list = $list;
        }
}

$foo1 = new Foo();

for($i = 0; $i < 1000000; $i++) {
        $a = array();
        if (!empty($a))
                $foo1->setList($a);
}
?>

使用命令运行该文件time。在一台特定的机器上,几次运行后平均需要 0.60 秒。注释掉if (!empty($a))导致它平均需要 3.00 秒才能运行的原因。

澄清:这些是例子。第一个示例演示了可怕的异常处理和可能的 DRY 违规,代价是一个简单的、非特定于域的示例。

4

4 回答 4

8

PHP 函数调用开销正好是 15.5355%。

:) 只是搅拌锅。

说真的,这里有几个关于这个主题的很棒的链接:

PHP 应用程序中是否可能有太多函数?

函数 vs 重复代码

这些链接上的代码可维护性与速度讨论解决了 OP 所暗示的(也许更重要的)问题,但只是为了添加一些数据,这些数据也可能是相关的,希望对将来遇到这个线程的人有用,这里是在 2011 Macbook Pro 上运行以下代码的结果(驱动器空间非常小,运行的程序太多)。

如其他地方所述,在决定是调用函数还是将代码“内联”时,一个重要的考虑因素是从某个代码块中调用该函数的次数。该函数被调用的次数越多,就越值得考虑进行内联工作。

结果(以秒为单位的时间)

调用函数方法 | 在线方法 | 差异 | 百分比差异

1,000 次迭代(4 次运行)

0.0039088726043701 | 0.0031478404998779 | 0.00076103210449219 | 19.4694

0.0038208961486816 | 0.0025999546051025 | 0.0012209415435791 | 31.9543

0.0030159950256348 | 0.0029480457305908 | 6.7949295043945E-5 | 2.2530

0.0031449794769287 | 0.0031390190124512 | 5.9604644775391E-6 | 0.1895

1,000,000 次迭代(4 次运行)

3.1843111515045 | 2.6896121501923 | 0.49469900131226 | 15.5355

3.131945848465 | 2.7114839553833 | 0.42046189308167 | 13.4249

3.0256152153015 | 2.7648048400879 | 0.26081037521362 | 8.6201

3.1251409053802 | 2.7397727966309 | 0.38536810874939 | 12.3312

function postgres_friendly_number($dirtyString) {
    
    $cleanString = str_ireplace("(", "-", $dirtyString);
    $badChars = array("$", ",", ")");
    $cleanString = str_ireplace($badChars, "", $cleanString);
    
    return $cleanString;
    
}


//main
$badNumberString = '-$590,832.61';

$iterations = 1000000;

$startTime = microtime(true);
for ($i = 1; $i <= $iterations; $i++) {
    $goodNumberString = postgres_friendly_number($badNumberString);
}
$endTime = microtime(true);
$firstTime = ($endTime - $startTime); 

$startTime = microtime(true);
for ($i = 1; $i <= $iterations; $i++) {
    $goodNumberString = str_ireplace("(", "-", $badNumberString);
    $badChars = array("$", ",", ")");
    $goodNumberString = str_ireplace($badChars, "", $goodNumberString);
}
$endTime = microtime(true); 
$secondTime = ($endTime - $startTime); 

$timeDifference = $firstTime - $secondTime;
$percentDifference = (( $timeDifference / $firstTime ) * 100);
于 2014-02-10T18:21:46.410 回答
8

还没有人讨论服务器的硬件与函数调用开销的关系。

调用函数时,所有 CPU 的寄存器都包含与当前执行点相关的数据。CPU 的所有寄存器都必须保存到内存中(通常是进程的堆栈),否则就没有希望返回到该执行点并恢复执行。从函数返回时,必须从内存(通常是进程的堆栈)中恢复所有 CPU 的寄存器。

因此,可以看到一串嵌套函数调用如何增加进程的开销。CPU 的寄存器必须一遍又一遍地保存在堆栈中,并一遍又一遍地恢复才能从函数中取回。

这确实是函数调用开销的来源。如果传递了函数参数,则必须在调用函数之前将它们全部复制。因此,将巨大的数组作为函数参数传递是一个糟糕的设计。

已经针对使用 getter/setter 的开销对面向对象的 PHP 进行了研究。删除所有 getter/setter 可将执行时间缩短约 50%。这仅仅是由于函数调用开销。

于 2014-11-18T21:40:12.053 回答
5

规范的 PHP 实现非常慢,因为它很容易实现,而且 PHP 所针对的应用程序不需要像快速函数调用这样的原始性能。

您可能需要考虑其他PHP 实现

如果您正在编写应该用 PHP 编写的应用程序(通过网络将数据从数据库转储到浏览器),那么函数调用开销并不大。当然不要特意去复制代码,因为你害怕使用一个函数会产生太多的开销。

于 2013-07-31T13:51:18.237 回答
4

我你混淆了术语。通常,函数调用开销是指调用从函数返回的开销。而不是内联处理。它不是函数调用的总成本。它只是准备参数和返回值以及执行分支的成本。

您遇到的问题是 PHP 和其他弱类型脚本风格的语言在确定函数是否有副作用方面非常糟糕。因此,它们不会将函数的结果存储为临时文件,而是会进行多次调用。如果函数正在做一些复杂的事情,这将是非常低效的。

所以,底线是:调用一次函数并存储和重用结果!不要使用相同的参数多次调用相同的函数。(没有充分的理由)

于 2014-08-12T01:00:51.400 回答