gpt4 book ai didi

php - 在 PHP 中显示已处理的脚本计算步骤

转载 作者:可可西里 更新时间:2023-11-01 00:28:46 24 4
gpt4 key购买 nike

是否可以显示 php 如何计算问题的解决方案。

例如:

<?php
$equation = (5+1)*3/9; // outputs 2
?>

如果想查看它处理的所有步骤以得出该解决方案:即:

(5+1) = 6

6*3 = 18

18/9 = 2

最佳答案

The short answer is "maybe". It's not possible to hook into PHP internals during runtime like you would have expected it. But, there is an alternative which comes close to it using an external parser.

PHP 脚本解析器和抽象语法树 (AST)

这样做的原因是,PHP 在内部使用抽象语法树 (AST) 来定义处理步骤,并通过创建更好的二进制表示作为 OpCache 来优化执行,以避免再次解析和处理相同的脚本又一次。

因此,如果您需要了解更多关于 PHP 中的处理,您可以考虑分析 PHP C 源代码:

PHP 脚本中的 AST - nikic/php-parser

也可以使用 PHP 重新实现 AST 解析器,请参阅 https://github.com/nikic/PHP-Parser .

以下示例将表达式 $equation = (5+1)*3/9;//将原始问题的 2 输出到专用的 math.php 文件。解析器使用以下 CLI 命令执行:

composer require nikic/php-parser
vendor/bin/php-parse math.php

这将输出 $equation = (5+1)*3/9; 的解析语法树;//输出 2(这基本上是对原始问题的回答):

====> File math.php:
==> Node dump:
array(
0: Stmt_Expression(
expr: Expr_Assign(
var: Expr_Variable(
name: equation
)
expr: Expr_BinaryOp_Div(
left: Expr_BinaryOp_Mul(
left: Expr_BinaryOp_Plus(
left: Scalar_LNumber(
value: 5
)
right: Scalar_LNumber(
value: 1
)
)
right: Scalar_LNumber(
value: 3
)
)
right: Scalar_LNumber(
value: 9
)
)
)
)
1: Stmt_Nop(
comments: array(
0: // outputs 2
)
)
)

OpCache 分析

通过阅读https://nikic.github.io/2017/04/14/PHP-7-Virtual-machine.html (由上面提到的 PHP AST 解析器的作者提供)可能会找到更多有关分析 PHP 处理的信息。但是,本节没有给出问题的额外答案 - 它更多地是概述备选方案以及它们实际揭示的内容。

php -d opcache.enable_cli=1 -d opcache.opt_debug_level=0x10000 math.php

上面在 CLI 上下文中执行的命令启用 OpCache 调试并输出以下内容:

$_main: ; (lines=3, args=0, vars=1, tmps=1)
; (before optimizer)
; /var/www/developer/math.php:1-4
L0: EXT_STMT
L1: ASSIGN CV0($equation) int(2)
L2: RETURN int(1)

上面的调试输出只包含结果 int(2) 但不包含已为语法树确定的相应步骤 - 这很明显,因为优化版本不需要它们。

这也可以通过查看生成的 OpCache 二进制文件来验证:

php -d opcache.enable_cli=1 -d opcache.opt_debug_level=0x10000 \
-d opcache.file_cache=/var/www/developer/opcache/ \
-d opcache.file_cache_only=1 math.php
hexdump opcache/08202de11af2c60edca0b5438eeefab6/var/www/developer/math.php.bin -C

(我的工作目录是/var/www/developer/,其中有一个子目录opcache/)

00000210  2f 76 61 72 2f 77 77 77  2f 64 65 76 65 6c 6f 70  |/var/www/develop|
00000220 65 72 2f 6d 61 74 68 2e 70 68 70 00 75 7f 00 00 |er/math.php.u...|
00000230 28 32 60 aa 75 7f 00 00 30 32 60 aa 75 7f 00 00 |(2`.u...02`.u...|
00000240 38 32 60 aa 75 7f 00 00 40 32 60 aa 75 7f 00 00 |82`.u...@2`.u...|
00000250 [02]00 00 00 00 00 00 00 04 00 00 00 ff ff ff ff |................| <--
00000260 01 00 00 00 00 00 00 00 04 00 00 00 ff ff ff ff |................|
00000270 e9 0b 00 00 00 00 00 00 00 00 00 00 00 00 00 00 |................|
00000280 00 00 00 00 00 00 00 00 02 00 00 00 65 08 08 08 |............e...|
00000290 e9 0b 00 00 00 00 00 00 50 00 00 00 00 00 00 00 |........P.......|
000002a0 00 00 00 00 00 00 00 00 02 00 00 00 26 10 01 08 |............&...|
000002b0 e9 0b 00 00 00 00 00 00 10 00 00 00 00 00 00 00 |................|
000002c0 00 00 00 00 ff ff ff ff 04 00 00 00 3e 01 08 08 |............>...|
000002d0 88 02 00 00 00 00 00 00 00 00 00 00 06 02 00 00 |................|
000002e0 cb 5b 29 c4 00 e7 1a 80 08 00 00 00 00 00 00 00 |.[).............|
000002f0 65 71 75 61 74 69 6f 6e 00 32 60 aa 75 7f 00 00 |equation.2`.u...|

上面的 hexdump 只是显示了 OpCache 文件的结尾。标记行中的第一个字节(位于 0x00000250)已包含优化结果 02。对于给定的示例,没有进一步指向运算符的指针(例如 ZEND_MUL),也没有整数文字 5139。因此,常数方程的结果已经直接存储在OpCache中。

关于php - 在 PHP 中显示已处理的脚本计算步骤,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49583825/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com