In diesem Artikel geht es ausführlich um die Mitgliedsattribute und -methoden von PHP-Klassen.
Im vorherigen Artikel haben wir die Funktion zend_do_begin_class_declaration vorgestellt, die zum Erstellen und Initialisieren eines zend_class_entry verwendet wird
Alle Informationen der Die Klasse wird in dieser Struktur gespeichert. Wie werden die Eigenschaften und Methoden
1
2
3 classPerson{ public$name;} Erinnern Sie sich auch an die im vorherigen Artikel erwähnte Funktion zend_initialize_class_data? Es spielt keine Rolle, ob Sie sich diese Funktion genauer ansehenzend_initialize_class_data(new_class_entry, 1 TSRMLS_CC);
234567891011121314
15161718192021222324252627282930 3132333435 3637383940414243 4445464748495051
5253545556575859606162 ZEND_APIvoidzend_initialize_class_data(zend_class_entry *ce , zend_bool nullify_handlers TSRMLS_DC)/ * {{ { */{ zend_bool persistent_hashes = (ce->type == ZEND_INTERNAL_CLASS) ? 1 : 0; s)? ; ce->refcount = 1; ce->constants_updated = 0; ce->ce_fla gs = 0 ; ash_init_ex(&ce ->default_properties, 0, NULL, zval_ptr_dtor_func, persistent_hashes, 0); zend_hash_init_ex(&ce->properties_info, 0, NULL, (dtor_func_t) (persistent_hashes ? ternal : zend_destroy_property_info ), persistent_hashes, 0); zend_hash_init_ex(&ce->default_static_members, 0, NULL, zval_ptr_dtor_func, persistent_hashes, 0); zend_hash_init_ex(&ce->function_table, 0, NULL , ZEND_FUNCTION_DTOR, persistent_hashes, 0); ifdef ZTS intn = zend_hash_num_elements(CG(class_table)); if(CG(static_members) && n >= CG (last_static_member)) {
/* Unterstützung für Laufzeitdeklaration: dl() CG(static_members) =realloc(CG(static_members), (n+1)*sizeof(HashTable*)); CG(static_members)[n] = NULL; } ce->static_members = (HashTable *)(zend_intptr_t)n; #else ce->static_members = NULL 🎜> ce->static_members = &ce->default_static_members;
if(nullify_handlers) {
ce->constructor = NULL;
ce->destructor =. NULL;
ce->clone = NULL;
ce->__get = NULL;
ce->__set = NULL;
ce->__unset = NULL;
ce- >__isset = NULL;
ce->__call = NULL;
ce->__callstatic =. ce->__tostring = NULL;
ce->create_object = NULL;
ce->get_iterator = NULL;
ce->iterator_funcs.funcs. = NULL;
ce- >interface_gets_implemented = NULL;
ce->get_static_method = NULL;
ce->parent = ce->num_interfaces = 0;
ce->interfaces = NULL;
ce->module = NULL;
ce->serialize =. NULL;
ce-> unserialize = NULL;
ce->serialize_func = NULL;
ce->unserialize_func =. NULL;
ce->builtin_functions = NULL;
1 : 0;
普通用户类与内部类..^.^注意看13-16行.
zend_hash_init_ex(&ce->default_properties, 0, NULL, zval_ptr_dtor_func, persistent_hashes, 0);zend_hash_init_ex(&ce->default_static_members, 0, NULL, zval_ptr_dtor_func, persistent_hashes, 0);
zend_hash_init_ex(&ce- >constants_table, 0, NULL, zval_ptr_dtor_func, persistent_hashes, 0);zend_hash_init_ex(&ce->function_table, 0, NULL, ZEND_FUNCTION_DTOR, persistent_hashes, 0);
如果你看过之前的文章,那么你定知道这是在初始化HashTable.是的..确实是这样,
default_properties,default_static_members等都是HashTable类型的指针.所以初始化当然要zend_hash_init了.第36-61行初始化魔术方法
不过这里只是初始化哦..好像并没有设置属性.$name属性是如何添加到属性表里的呢???
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
{ zend_do_begin_class_declaration(&$1, &$2, &$3 TSRMLS_CC); }
implements_list
'{'
class_statement_list
'}'{ zend_do_end_class_declaration(&$1, &$2 TSRMLS_CC); }
| interface_entry T_STRING
{ zend_do_begin_class_declaration(&$1, &$2, NULL TSRMLS_CC); }
interface_extends_list
'{'
class_statement_list
'}'{ zend_do_end_class_declaration(&$1, &$2 TSRMLS_CC); }
;
class_statement:
variable_modifiers { CG(access_type) = Z_LVAL($1.u.constant); } class_variable_declaration';'
| class_constant_declaration';'
| method_modifiers function is_reference T_STRING { zend_do_begin_function_declaration(&$2, &$4, 1, $3.op_type, &$1 TSRMLS_CC); }'('
parameter_list')'method_body { zend_do_abstract_method(&$4, &$1, &$9 TSRMLS_CC); zend_do_end_function_declaration(&$2 TSRMLS_CC); }
;
class_variable_declaration:
_property(&$3, NULL, CG(access_type) TSRMLS_CC); }
| class_variable_declaration','T_VARIABLE'='static_scalar { zend_do_declare_property(&$3, &$5, CG(access_type) TSRMLS_CC); }
| T_VARIABLE { zend_do_declare_property(&$1, NULL, CG(access_type) TSRMLS_CC); }
| T_VARIABLE'='static_scalar { zend_do_declare_property(&$1, &$3, CG(access_type) TSRMLS_CC); 🎜> 类体里会调用 zend_do_declare_property处理.
12
3
121314151617181920212223242526272829303132 333435363738
39
40
41
42
43
44
voidzend_do_declare_property(constznode *var_name,constznode *value, zend_uint access_type TSRMLS_DC)/* {{{ */
{
zval *property;
zend_property_info. *ex isting_property_info;
char*comment = NULL;
intcomment_len = 0;
if(CG(active_class_entry)->ce_flags & ZEND_ACC_INTERFACE) {
🎜> zend_error (E_COMPILE_ERROR,"Eigenschaften können nicht als abstrakt deklariert werden"); } if(access_type & ZEND_ACC_FINAL) { zend_error(E_COM PILE_ERROR,"Kann nicht Eigenschaft %s::$%s final deklarieren, der finale Modifikator ist nur für Methoden und Klassen zulässig", CG(active_class_entry)->name, var_name->u.constant.value.str. val); } if(zend_hash_find(&CG(active_class_entry)->properties_info, var_name->u.constant.value.str.val, var_name ->u.constant.value.str.len+1, (void**) &existing_property_info)==SUCCESS) { if(!(existing_property_info->flags & ZEND_ACC_IMPLICIT_PUBLIC)) {} ;u. Konstante; }else{ INIT_PZVAL(property); Z_TYPE_P(property) =. IS_NULL; } if(CG(doc_comment)) { comment = CG(doc_comment); comment_len =. CG(doc_comment_len); CG(doc_comment) = NULL ; CG(doc_comment_len) = 0; u.konstanter.Wert. str.val, var_name->u.constant.value.str.len, property, access_type, comment, comment_len TSRMLS_CC); efree(var_name->u.constant.value.str.val) ;}
Zeile 8-25:
Wenn Ihre Klasse eine Schnittstelle deklariert, kann die Schnittstelle keine Attribute haben und wird ausgelöst. Schnittstellen dürfen keine Mitgliedsvariablen enthalten.
Wenn die Attribute der Klasse auf abstrakt gesetzt sind, dann wird dies der Fall sein throw Eigenschaften können nicht als abstrakt deklariert werden
Wenn die Eigenschaften der Klasse auf „final“ gesetzt sind, wird die Eigenschaft „%s::$%s final“ kann nicht deklariert werden, der finale Modifikator ist nur für Methoden und Klassen zulässig
Alles ist OK, es werden ZVAL-Daten zugewiesen. Wenn das Attribut einen Anfangswert hat, werden die Daten dem ZVAL zugewiesen. Wenn nicht, wird INIT_PZVAL aufgerufen, um das ZVAL zu initialisieren, und der Typ wird auf IS_NULL gesetzt.
Schließlich wird zend_declare_property_ex aufgerufen. Fügen Sie den zval zum angegebenen active_class_entry hinzu.
Methoden der Klasse
Schauen Sie sich zuerst die Regeln an
Kommt Ihnen zend_do_begin_function_declaration bekannt vor?
Wenn Sie das vorherige gelesen haben Der Artikel kommt Ihnen bekannt vorWenn Sie ihn noch nicht gelesen haben, lesen Sie zuerst diesen Artikel. Die Definition der Funktion
Ich werde hier nicht auf Details eingehen.Ich werde nur darüber sprechen, was war in diesem Artikel nicht erwähnt
In dieser Funktion gibt es ein Urteil in
1
wird geworfen und dann
if (zend_hash_add() wird aufgerufen &CG(active_class_entry)->function_table, lcname, name_len+1, &op_array, sizeof(zend_op_array), (void **) &CG(active_op_array)) == FAILURE) {zend_error(E_COMPILE_ERROR, „Cannot redeclare %s: :%s()“, CG(active_class_entry)->name, name);
}Fügen Sie die Methode direkt zur Funktionstabelle hinzu.
Im Folgenden werden unterschiedliche Urteile basierend auf unterschiedlichen Klassendeklarationen getroffen. Das Obige ist der Originalinhalt: PHP-Kernelforschung und andere Mitgliedsattribute und -methoden. Weitere verwandte Inhalte finden Sie auf der chinesischen PHP-Website (www.php.cn)!