- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
假设我们有两个 php 文件,a.php 和 b.php
这是文件 a.php 的内容:
<?php // content of a.php
class A {
}
<?php // content of b.php
include dirname(__FILE__) . "/a.php";
echo "A: ", class_exists("A") ? "exists" : "doesn’t exist", "\n";
echo "B: ", class_exists("B") ? "exists" : "doesn’t exist", "\n";
echo "BA (before): ", class_exists("BA") ? "exists" : "doesn’t exist", "\n";
echo "BB: ", class_exists("BB") ? "exists" : "doesn’t exist", "\n";
class B {
}
class BA extends A {
}
class BB extends B {
}
echo "BA (after): ", class_exists("BA") ? "exists" : "doesn’t exist", "\n";
A: exists
B: exists
BA (before): doesn’t exist
BB: exists
BA (after): exists
最佳答案
免责声明:我并不声称了解 Zend 的内部工作原理。以下是我对 PHP 源代码的解释,很大程度上是由有根据的猜测插入的。尽管我对结论充满信心,但术语或细节可能会偏离。我很想听听任何在 Zend 内部有经验的人对此事的看法。
调查
来自 PHP 解析器 we can see当遇到类声明时 zend_do_early_binding
函数被调用。 Here是处理派生类声明的代码:
case ZEND_DECLARE_INHERITED_CLASS:
{
zend_op *fetch_class_opline = opline-1;
zval *parent_name;
zend_class_entry **pce;
parent_name = &CONSTANT(fetch_class_opline->op2.constant);
if ((zend_lookup_class(Z_STRVAL_P(parent_name), Z_STRLEN_P(parent_name), &pce TSRMLS_CC) == FAILURE) ||
((CG(compiler_options) & ZEND_COMPILE_IGNORE_INTERNAL_CLASSES) &&
((*pce)->type == ZEND_INTERNAL_CLASS))) {
if (CG(compiler_options) & ZEND_COMPILE_DELAYED_BINDING) {
zend_uint *opline_num = &CG(active_op_array)->early_binding;
while (*opline_num != -1) {
opline_num = &CG(active_op_array)->opcodes[*opline_num].result.opline_num;
}
*opline_num = opline - CG(active_op_array)->opcodes;
opline->opcode = ZEND_DECLARE_INHERITED_CLASS_DELAYED;
opline->result_type = IS_UNUSED;
opline->result.opline_num = -1;
}
return;
}
if (do_bind_inherited_class(CG(active_op_array), opline, CG(class_table), *pce, 1 TSRMLS_CC) == NULL) {
return;
}
/* clear unnecessary ZEND_FETCH_CLASS opcode */
zend_del_literal(CG(active_op_array), fetch_class_opline->op2.constant);
MAKE_NOP(fetch_class_opline);
table = CG(class_table);
break;
}
zend_lookup_class
查看符号表中是否存在父类......然后根据是否找到父类而发散。
if (do_bind_inherited_class(CG(active_op_array), opline, CG(class_table), *pce, 1 TSRMLS_CC) == NULL) {
return;
}
do_bind_inherited_class
,我们看到最后一个参数(在这个调用中是
1
)被称为
compile_time
.这听起来很有趣。这个论点有什么作用?
if (compile_time) {
op1 = &CONSTANT_EX(op_array, opline->op1.constant);
op2 = &CONSTANT_EX(op_array, opline->op2.constant);
} else {
op1 = opline->op1.zv;
op2 = opline->op2.zv;
}
found_ce = zend_hash_quick_find(class_table, Z_STRVAL_P(op1), Z_STRLEN_P(op1), Z_HASH_P(op1), (void **) &pce);
if (found_ce == FAILURE) {
if (!compile_time) {
/* If we're in compile time, in practice, it's quite possible
* that we'll never reach this class declaration at runtime,
* so we shut up about it. This allows the if (!defined('FOO')) { return; }
* approach to work.
*/
zend_error(E_COMPILE_ERROR, "Cannot redeclare class %s", Z_STRVAL_P(op2));
}
return NULL;
} else {
ce = *pce;
}
compile_time
地位。然后它尝试在类表中找到类条目(“ce”),如果没有找到,那么......它在编译时不做任何事情就返回,但在运行时发出 fatal error 。
zend_do_early_binding
.如果找不到父类怎么办?
if (CG(compiler_options) & ZEND_COMPILE_DELAYED_BINDING) {
zend_uint *opline_num = &CG(active_op_array)->early_binding;
while (*opline_num != -1) {
opline_num = &CG(active_op_array)->opcodes[*opline_num].result.opline_num;
}
*opline_num = opline - CG(active_op_array)->opcodes;
opline->opcode = ZEND_DECLARE_INHERITED_CLASS_DELAYED;
opline->result_type = IS_UNUSED;
opline->result.opline_num = -1;
}
return;
do_bind_inherited_class
再次——但这一次,
compile_time
的值将是
0
(错误的)。
class_exists
PHP函数?查看源代码显示了这个片段:
found = zend_hash_find(EG(class_table), name, len+1, (void **) &ce);
class_table
变量相同
class_table
参与
do_bind_inherited_class
调用我们之前看到的!所以返回值
class_exists
取决于该类的条目是否已插入
class_table
来自
do_bind_inherited_class
.
include
编译时的指令(即使文件名是硬编码的)。
compile_time
发出类重新声明 fatal error 。未设置标志;错误可以无条件地发出。
ZEND_DECLARE_INHERITED_CLASS_DELAYED
执行脚本时注册类的操作码。那时
compile_time
标志将是
false
和行为会略有不同。
class_exists
的返回值取决于类是否已经注册。
class_exists
的行为也不一样:
class_exists
返回 false
,实例化会产生 fatal error )关于php - 稍后在同一文件 "does not exist"中定义的派生类?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12617188/
我正在编写一个插件,有一个ajax调用向用户显示数据。 如果用户想在ajax成功时添加一些js?他可以从他的 js 脚本中做到这一点吗,比如定位这个 ajax 成功事件。 例如: $(documen
我有 html 代码,例如 - x 最初插入 div 'insert_calendar_eda_form'。 Javascript代码 calendar_eda_add
关闭。这个问题需要更多focused .它目前不接受答案。 想改进这个问题吗? 更新问题,使其只关注一个问题 editing this post . 关闭 3 年前。 Improve this qu
我已经使用命令 sudo start myservice 启动了一个 upstart 服务。我想要一种方法,以便稍后我(或任何人)可以检查该服务是否已启动并正在运行。检查它的命令是什么? 最佳答案 找
我是一名优秀的程序员,十分优秀!