>  기사  >  백엔드 개발  >  PHP 소스 코드 28에 대한 간략한 토론: 클래스 구조 및 상속에 대하여

PHP 소스 코드 28에 대한 간략한 토론: 클래스 구조 및 상속에 대하여

不言
不言원래의
2018-06-29 09:24:011802검색

이 글은 주로 PHP 소스 코드에 대한 28가지 단어를 소개합니다: 클래스 구조와 상속에 관해서는 특정 참조 가치가 있습니다. 이제 필요한 친구들이 참조할 수 있습니다.

#🎜 🎜 # PHP 소스 코드 28에 대한 간략한 설명: 클래스 구조 및 상속에 대하여

객체 지향에서 매우 중요하고 복잡하게 얽힌 기능으로서 우리는 뭔가를 이해해야 합니다

PHP5에서는, 상속의 개념과 함께 오늘은 PHP 소스 코드부터 시작하여 상속이 어떻게 구현되는지 이해합니다.
클래스 상속을 이해하기 전에 PHP 소스 코드에서 클래스가 어떻게 저장되는지 알아야 합니다.
zend/zend.h의 418번째 줄을 찾으세요:

 struct _zend_class_entry {
char type;
char *name;/* 类名 */
zend_uint name_length;/* 类名字符串长度 */
struct _zend_class_entry *parent; /* 父类 */
int refcount; /* 引用计数 */
zend_bool constants_updated;
zend_uint ce_flags;/* 类的访问控制 */ 
HashTable function_table;/*类的成员函数 */
HashTable default_properties;/*类的默认属性 */
HashTable properties_info;/*类的属性信息 如访问控制等 */
HashTable default_static_members;/*静态成员列表 */
HashTable *static_members;
HashTable constants_table;/* 常量列表 */
const struct _zend_function_entry *builtin_functions; union _zend_function *constructor;/*构造函数*/
union _zend_function *destructor;/*析构函数*/
union _zend_function *clone;/*克隆方法*/ /*魔术方法 */
union _zend_function *__get;
union _zend_function *__set;
union _zend_function *__unset;
union _zend_function *__isset;
union _zend_function *__call;
union _zend_function *__callstatic;
union _zend_function *__tostring;
union _zend_function *serialize_func;
union _zend_function *unserialize_func; 
zend_class_iterator_funcs iterator_funcs; /* handlers */
zend_object_value (*create_object)(zend_class_entry *class_type TSRMLS_DC);
zend_object_iterator *(*get_iterator)(zend_class_entry *ce, zval *object, int by_ref TSRMLS_DC);
int (*interface_gets_implemented)(zend_class_entry *iface, zend_class_entry *class_type TSRMLS_DC); /* a class implements this interface */
union _zend_function *(*get_static_method)(zend_class_entry *ce, char* method, int method_len TSRMLS_DC); /* serializer callbacks */
int (*serialize)(zval *object, unsigned char **buffer, zend_uint *buf_len, zend_serialize_data *data TSRMLS_DC);
int (*unserialize)(zval **object, zend_class_entry *ce, const unsigned char *buf, zend_uint buf_len, zend_unserialize_data *data TSRMLS_DC); 
zend_class_entry **interfaces;/*类实现的接口 */
zend_uint num_interfaces;/*类实现的接口数 */ /*类所在文件信息 */
char *filename;
zend_uint line_start;
zend_uint line_end;
char *doc_comment;
zend_uint doc_comment_len; struct _zend_module_entry *module;};

PHP와 함께 제공되는 SPL에서 클래스 상속을 볼 수 있습니다. 여기에서 필요한 구현을 찾을 수 있습니다.

매뉴얼에서 CachingIterator 클래스를 진입점으로 선택하세요.

CachingIterator 확장 IteratorIterator는 OuterIterator, Traversable, Iterator, ArrayAccess, Countable을 구현합니다.
추적 프로세스는 다음과 같습니다.

 //在ext/spl/spl_iterators.c 3246行,REGISTER_SPL_SUB_CLASS_EX(CachingIterator, IteratorIterator, spl_dual_it_new, spl_funcs_CachingIterator); //    ext/spl/spl_functions.h 34行#define REGISTER_SPL_SUB_CLASS_EX(class_name, parent_class_name, obj_ctor, funcs) \
    spl_register_sub_class(&spl_ce_ ## class_name, spl_ce_ ## parent_class_name, # class_name, obj_ctor, funcs TSRMLS_CC); //    ext/spl/spl_functions.c 56行PHPAPI void spl_register_sub_class(zend_class_entry ** ppce, zend_class_entry * parent_ce, char * class_name, void *obj_ctor, const zend_function_entry * function_list TSRMLS_DC)*ppce = zend_register_internal_class_ex(&ce, parent_ce, NULL TSRMLS_CC); //    zend/zend_API.c  2196行ZEND_API zend_class_entry *zend_register_internal_class_ex(zend_class_entry *class_entry, zend_class_entry *parent_ce, char *parent_name TSRMLS_DC)zend_do_inheritance(register_class, parent_ce TSRMLS_CC); //    zend/zend_compile.c 2784行 
ZEND_API void zend_do_inheritance(zend_class_entry *ce, zend_class_entry *parent_ce TSRMLS_DC){
    /* 接口不能继承类 */
    if ((ce->ce_flags & ZEND_ACC_INTERFACE)
    && !(parent_ce->ce_flags & ZEND_ACC_INTERFACE)) {
    zend_error(E_COMPILE_ERROR, "Interface %s may not inherit from class (%s)", ce->name, parent_ce->name);
    }     /* final类不能继承 */
    if (parent_ce->ce_flags & ZEND_ACC_FINAL_CLASS) {
    zend_error(E_COMPILE_ERROR, "Class %s may not inherit from final class (%s)", ce->name, parent_ce->name);
    } 
    ce->parent = parent_ce;
    /* 拷贝序列化和反序列化回调函数 */
    if (!ce->serialize) {
    ce->serialize   = parent_ce->serialize;
    }
    if (!ce->unserialize) {
    ce->unserialize = parent_ce->unserialize;
    }     /* 继承父类的接口 在此处有停下来思考,为何要这么做 */
    zend_do_inherit_interfaces(ce, parent_ce TSRMLS_CC);     /* 继承父类的属性  */
    zend_hash_merge(&ce->default_properties, &parent_ce->default_properties, (void (*)(void *)) zval_add_ref, NULL, sizeof(zval *), 0);
    if (parent_ce->type != ce->type) {
    /* 用户自定义的类从内部类继承 */
    zend_update_class_constants(parent_ce  TSRMLS_CC);
    zend_hash_apply_with_arguments(CE_STATIC_MEMBERS(parent_ce) TSRMLS_CC, (apply_func_args_t)inherit_static_prop, 1, &ce->default_static_members);
    } else {
    zend_hash_apply_with_arguments(&parent_ce->default_static_members TSRMLS_CC, (apply_func_args_t)inherit_static_prop, 1, &ce->default_static_members);
    }
    zend_hash_merge_ex(&ce->properties_info, &parent_ce->properties_info, (copy_ctor_func_t) (ce->type & ZEND_INTERNAL_CLASS ? zend_duplicate_property_info_internal : zend_duplicate_property_info), sizeof(zend_property_info), (merge_checker_func_t) do_inherit_property_access_check, ce);     /* 合并常量和成员函数 */
    zend_hash_merge(&ce->constants_table, &parent_ce->constants_table, (void (*)(void *)) zval_add_ref, NULL, sizeof(zval *), 0);
    zend_hash_merge_ex(&ce->function_table, &parent_ce->function_table, (copy_ctor_func_t) do_inherit_method, sizeof(zend_function), (merge_checker_func_t) do_inherit_method_check, ce); 
    do_inherit_parent_constructor(ce);     if (ce->ce_flags & ZEND_ACC_IMPLICIT_ABSTRACT_CLASS && ce->type == ZEND_INTERNAL_CLASS) {
    ce->ce_flags |= ZEND_ACC_EXPLICIT_ABSTRACT_CLASS;
    } else if (!(ce->ce_flags & ZEND_ACC_IMPLEMENT_INTERFACES)) {
    /* The verification will be done in runtime by ZEND_VERIFY_ABSTRACT_CLASS */
    zend_verify_abstract_class(ce TSRMLS_CC);
    }}

위의 do_inherit_parent_constructor(ce)에 대하여

#🎜 🎜#이 메소드는 매직 메소드 상속을 구현합니다. 하위 클래스에 해당 매직 메소드가 없으면 상위 클래스의 해당 메소드가 상속됩니다. 아래 표시된 PHP 코드는 하위 클래스에 생성자가 없는 경우입니다

class Base {
public function __construct() {
    echo &#39;Base __construct<br />&#39;;
}} class Foo extends Base { } $foo = new Foo();

위와 같이 다음과 같이 출력됩니다. Base __construct

하위 클래스에 생성자가 있는 경우 분명히 상위 클래스의 생성자를 상속합니다. 자신의 생성자 메소드를 사용하며 상위 클래스의 생성자를 호출해야 할 경우 하위 클래스의 생성자에서 상위 클래스의 생성자를 호출해야 하며 PHP는 이를 자동으로 호출하지 않습니다.


위 내용은 모두의 학습에 도움이 되기를 바랍니다. 더 많은 관련 내용은 PHP 중국어 홈페이지를 주목해주세요!

관련 권장 사항:

PHP 소스 코드 27에 대한 간략한 토론: PHP의 구성 방법 식별

#🎜🎜 ## 🎜🎜# PHP 소스 코드 26에 대한 간략한 토론: PHP 빠른 정렬 소스 코드 구현 단순화

간략한 내용 PHP 소스 코드 25에 대한 토론: 다음, 현재, 주요 함수에 대하여

위 내용은 PHP 소스 코드 28에 대한 간략한 토론: 클래스 구조 및 상속에 대하여의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.