PHP 8,函数别名兼容性`getdir()` 问题

如何解决PHP 8,函数别名兼容性`getdir()` 问题

在测试我的 脚本是否与 兼容时,我遇到了以下代码:

function getDir($a,$o = 2) {
    $d = Floor($a / $o);
    return ($d % 2 === 0);
}

之前,这运行良好,但是,在 上它抛出:

致命错误:无法重新声明 getDir()

3v4l.org


搜索一段时间后,我发现 dir() 引入了一个新别名:

/** @param resource $context */
function getdir(string $directory,$context = null): Directory|false {}

php-src line 709

问题

解决方法

简答:哎呀

长答案:https://externals.io/message/113982

目前,我计划在 8.0.5 之前将其删除。抱歉打扰了,感谢 Chris 提交关于此的错误报告:https://bugs.php.net/bug.php?id=80914

,

事实证明这比我预期的要有趣得多。

简短的回答是 getdir() 在 PHP 8.0.0 中确实是新的,但这是一个错误,它可能会在 8.0.4 或 8.0.5 中被删除。

有趣的是,getdir() 实际上并不是别名,而是函数内部的真实名称;只是在 8.0 之前,它只能通过其别名 dir() 访问。为了解释这一点,我们必须追溯到 20 多年前...


dir() 函数是在 PHP 3.0 中添加的。无论出于何种原因——也许是在最后一刻更改了名称——实现它的 C 函数被称为“php3_getdir”而不是“php3_dir”。这并不重要,因为每个函数名称都被显式映射,如下所示:

function_entry php3_dir_functions[] = {
    {"opendir",php3_opendir,NULL},{"closedir",php3_closedir,{"chdir",php3_chdir,{"rewinddir",php3_rewinddir,{"readdir",php3_readdir,{"dir",php3_getdir,{NULL,NULL,NULL}
};

不久之后,PHP 4 出现了,函数定义 moved to using macros 将 C 名称与 PHP 名称相匹配。由于函数和实现的名称不匹配,“dir”最终被标记为“别名”;但没有为“getdir”添加额外的条目:

static zend_function_entry php_dir_functions[] = {
    PHP_FE(opendir,NULL)
    PHP_FE(closedir,NULL)
    PHP_FE(chdir,NULL)
    PHP_FE(rewinddir,NULL)
    PHP_FE(readdir,NULL)
    PHP_FALIAS(dir,getdir,NULL)
    {NULL,NULL}
};

没有目标的别名实际上没有意义(并且有一个 PHP_NAMED_FE 宏就是为了这个目的)但它有效,所以我想没有人注意到。

事实上,通过 PHP 5 和 PHP 7 的所有变化,它继续工作,with basically the same line of C code right up to 7.4

PHP_FALIAS(dir,arginfo_dir)

然而,在 PHP 8 的工作期间,构建了一个系统来从 PHP“存根”生成内部函数信息。作为此 stubs were added for all function aliases 的一部分,getdir() 以 its own stub 结束:

/** @param resource $context */
function getdir(string $directory,$context = null): Directory|false {}

/**
 * @param resource|null $context
 * @alias getdir
 */
function dir(string $directory,$context = null): Directory|false {}

然后用它重新生成 C 定义,最后 getdir() had its own function entry

ZEND_FE(getdir,arginfo_getdir)
ZEND_FALIAS(dir,arginfo_dir)

这导致 getdir() 成为真正的内置函数名,这意味着您不能拥有同名的函数。


从那时起,发生了四件事:

  • 2021 年 3 月 29 日:0stone0 发布了这个问题。
  • Chris Haas 试图追查该问题,并打开 a bug on the PHP bug tracker,认为该问题与区分大小写有关。他们还确认 getdir() 是 basic_functions.stub.php 中唯一不在手册中的别名。
  • 2021 年 4 月 6 日:我以为我要快速修复文档,于是产生了兴趣,然后消失在了上面的兔子洞里。我posted my findings to the PHP internals mailing list比预期晚了一个小时上床睡觉。
  • Sara Golemon(PHP 8.0 发布经理之一)回复同意应将其视为错误并在下一个 8.0.x 版本中恢复。她也posted an answer here,因为我还没有。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


依赖报错 idea导入项目后依赖报错,解决方案:https://blog.csdn.net/weixin_42420249/article/details/81191861 依赖版本报错:更换其他版本 无法下载依赖可参考:https://blog.csdn.net/weixin_42628809/a
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下 2021-12-03 13:33:33.927 ERROR 7228 [ main] o.s.b.d.LoggingFailureAnalysisReporter : *************************** APPL
错误1:gradle项目控制台输出为乱码 # 解决方案:https://blog.csdn.net/weixin_43501566/article/details/112482302 # 在gradle-wrapper.properties 添加以下内容 org.gradle.jvmargs=-Df
错误还原:在查询的过程中,传入的workType为0时,该条件不起作用 <select id="xxx"> SELECT di.id, di.name, di.work_type, di.updated... <where> <if test=&qu
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct redisServer’没有名为‘server_cpulist’的成员 redisSetCpuAffinity(server.server_cpulist); ^ server.c: 在函数‘hasActiveC
解决方案1 1、改项目中.idea/workspace.xml配置文件,增加dynamic.classpath参数 2、搜索PropertiesComponent,添加如下 <property name="dynamic.classpath" value="tru
删除根组件app.vue中的默认代码后报错:Module Error (from ./node_modules/eslint-loader/index.js): 解决方案:关闭ESlint代码检测,在项目根目录创建vue.config.js,在文件中添加 module.exports = { lin
查看spark默认的python版本 [root@master day27]# pyspark /home/software/spark-2.3.4-bin-hadoop2.7/conf/spark-env.sh: line 2: /usr/local/hadoop/bin/hadoop: No s
使用本地python环境可以成功执行 import pandas as pd import matplotlib.pyplot as plt # 设置字体 plt.rcParams['font.sans-serif'] = ['SimHei'] # 能正确显示负号 p
错误1:Request method ‘DELETE‘ not supported 错误还原:controller层有一个接口,访问该接口时报错:Request method ‘DELETE‘ not supported 错误原因:没有接收到前端传入的参数,修改为如下 参考 错误2:cannot r
错误1:启动docker镜像时报错:Error response from daemon: driver failed programming external connectivity on endpoint quirky_allen 解决方法:重启docker -> systemctl r
错误1:private field ‘xxx‘ is never assigned 按Altʾnter快捷键,选择第2项 参考:https://blog.csdn.net/shi_hong_fei_hei/article/details/88814070 错误2:启动时报错,不能找到主启动类 #
报错如下,通过源不能下载,最后警告pip需升级版本 Requirement already satisfied: pip in c:\users\ychen\appdata\local\programs\python\python310\lib\site-packages (22.0.4) Coll
错误1:maven打包报错 错误还原:使用maven打包项目时报错如下 [ERROR] Failed to execute goal org.apache.maven.plugins:maven-resources-plugin:3.2.0:resources (default-resources)
错误1:服务调用时报错 服务消费者模块assess通过openFeign调用服务提供者模块hires 如下为服务提供者模块hires的控制层接口 @RestController @RequestMapping("/hires") public class FeignControl
错误1:运行项目后报如下错误 解决方案 报错2:Failed to execute goal org.apache.maven.plugins:maven-compiler-plugin:3.8.1:compile (default-compile) on project sb 解决方案:在pom.
参考 错误原因 过滤器或拦截器在生效时,redisTemplate还没有注入 解决方案:在注入容器时就生效 @Component //项目运行时就注入Spring容器 public class RedisBean { @Resource private RedisTemplate<String
使用vite构建项目报错 C:\Users\ychen\work>npm init @vitejs/app @vitejs/create-app is deprecated, use npm init vite instead C:\Users\ychen\AppData\Local\npm-