Heim >Backend-Entwicklung >PHP-Tutorial >Original: PHP-Kernel-Forschung zu Mitgliedsattributen und Methoden wie

Original: PHP-Kernel-Forschung zu Mitgliedsattributen und Methoden wie

黄舟
黄舟Original
2016-12-22 09:58:271307Durchsuche

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 ansehen

zend_initialize_class_data(new_class_entry, 1 TSRMLS_CC);

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

48

49

50

51

52

53

54

55

56

57

58

59

60

61

62

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处理.

1

2


3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

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

1

2

3

4

5

classPerson{

publicfunctiontest(){

echo1;

}

}

Was ist, wenn es sich um eine Methode handelt?

Schauen Sie sich zuerst die Regeln an

1

2

3

4

5

class_statement:

CG(access_type) = Z_LVAL($1.u.constant); } class_variable_declaration';'

|. gin_function_declaration(&$2, &$4, " RMLS_CC); }

Das erste ist das Attribut, dann das dritte die Methode...

Kommt Ihnen zend_do_begin_function_declaration bekannt vor?

Wenn Sie das vorherige gelesen haben Der Artikel kommt Ihnen bekannt vor

Wenn 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

2

3

4

5

6

7

8

9

10

11

if (is_method) {

if(CG(active_class_entry)->ce_flags & ZEND_ACC_INTERFACE) {

if((Z_LVAL(fn_flags_znode-&g t;u.constant) & ~(ZEND_ACC_STATIC|ZEND_ACC_PUBLIC) :) .str.val);

                                                                       Rest des Parsers */

}

                              fn_flags = Z_LVAL( fn_flags_znode->u.constant);                             ​​​::%s() muss weggelassen werden

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)!


Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn