0

0

PHP 面向对象编程:解决继承中遇到的 NULL 值问题与正确设计类关系

DDD

DDD

发布时间:2025-09-27 10:17:19

|

892人浏览过

|

来源于php中文网

原创

PHP 面向对象编程:解决继承中遇到的 NULL 值问题与正确设计类关系

本文深入探讨 PHP 面向对象编程中常见的 NULL 值问题,重点讲解了如何正确使用 __construct 方法作为类构造函数来初始化对象属性,以及如何区分和恰当应用类之间的继承(is-a)与聚合(has-a)关系。通过具体代码示例,指导开发者避免因构造函数误用或不当继承设计导致的运行时错误,优化代码结构和可维护性。

1. 问题剖析:为何出现 NULL 值?

php 面向对象编程中,当尝试创建对象并为其属性赋值时,如果操作不当,可能会导致属性值为 null。原始代码中,clinic 类在尝试通过 assignpatient 方法添加 patient 对象时,遇到了这个问题。

// 原始 Patient 类片段
class Patient{
    private $name;
    private $age;
    private $gender;

    public function record($name, $age, $gender){
        $this->name = $name;
        $this->age = $age;
        $this->gender = $gender;
    }
    // ...
}

// 原始 Clinic 类片段
class Clinic extends Patient{
    private $patients = [];

    public function assignPatient($name, $age, $gender){
        // 问题所在:这里调用了 new Patient()
        // 但 Patient 类中没有定义构造函数,record() 也未被调用
        $this->patients[] = new Patient($name, $age, $gender);
    }
    // ...
}

导致 NULL 值输出的主要原因有两点:

  1. 构造函数缺失或误用: 在 Patient 类中定义了一个名为 record 的方法来设置属性,但它并不是 PHP 的特殊方法 __construct。当通过 new Patient(...) 创建对象时,PHP 默认会寻找并执行 __construct 方法来初始化对象。如果 __construct 不存在,且没有其他方式在对象创建时显式调用 record 方法,那么 Patient 对象的 $name, $age, $gender 属性将保持其默认的 NULL 值。
  2. 不恰当的类继承关系: Clinic extends Patient 表达的是“诊所是一种病人”的“is-a”关系,这在逻辑上是不合理的。一个诊所通常是管理或包含病人的,而不是病人本身。这种不恰当的继承关系虽然不是导致 NULL 值的直接原因,但它混淆了类的职责,增加了代码的复杂性和理解难度。

2. PHP 构造函数 __construct 的正确使用

PHP 中的 __construct 是一个特殊的方法,被称为构造函数。当使用 new 关键字创建类的实例时,该方法会自动被调用。它的主要作用是初始化新创建的对象,例如设置属性的初始值、执行必要的设置逻辑等。

为了解决 Patient 对象属性为 NULL 的问题,我们需要将 record 方法重命名为 __construct,并确保它在对象创建时接收并设置所需的参数。

name = $name;
        $this->age = $age;
        $this->gender = $gender;
    }

    // 获取病人姓名
    public function getName(){
        return $this->name;
    }

    // 获取病人年龄
    public function getAge(){
        return $this->age;
    }

    // 获取病人性别
    public function getGender(){
        return $this->gender;
    }
}
?>

通过上述修改,现在当我们执行 new Patient("Patrick star", 18, "Male") 时,__construct 方法会自动执行,并将传入的姓名、年龄和性别赋值给 $name, $age, $gender 属性,确保对象被正确初始化。

立即学习PHP免费学习笔记(深入)”;

3. 理解类关系:继承与聚合

在面向对象设计中,正确地建立类之间的关系至关重要。常见的两种关系是继承(Inheritance)和聚合(Aggregation/Composition)。

3.1 继承(Inheritance):"is-a" 关系

继承表示一个类是另一个类的特殊类型。例如,“狗是一种动物”,那么 Dog 类可以继承 Animal 类。继承通过 extends 关键字实现。

  • 特点: 子类会继承父类的公共(public)和受保护(protected)的属性和方法。子类可以重写父类的方法,也可以添加自己的新属性和方法。
  • 适用场景: 当子类确实是父类的一种更具体的实现时。

在原始代码中,Clinic extends Patient 意味着“诊所是一种病人”。这显然不符合现实逻辑。一个诊所不是一个病人,它是一个管理病人的实体。因此,这种继承关系是不恰当的。

燕雀Logo
燕雀Logo

为用户提供LOGO免费设计在线生成服务

下载

3.2 聚合(Aggregation/Composition):"has-a" 关系

聚合表示一个类包含另一个类的实例作为其成员。例如,“诊所拥有病人”,那么 Clinic 类会包含一个或多个 Patient 对象的集合。聚合通过在一个类中声明另一个类的实例作为属性来实现。

  • 特点: 一个类作为另一个类的组成部分。被包含的类是独立存在的,也可以被其他类使用。
  • 适用场景: 当一个类需要使用另一个类的功能或数据,并且它们之间是“拥有”或“包含”的关系时。

对于 Clinic 和 Patient 的关系,更合理的模型是聚合:一个 Clinic 对象“拥有”一个或多个 Patient 对象的集合。因此,Clinic 类不应该继承 Patient,而应该在其内部维护一个 Patient 对象的数组。

基于此,我们重构 Clinic 类,移除不必要的继承,并使其通过聚合关系管理 Patient 对象:

patients;
    }

    /**
     * 向诊所添加一位新病人。
     *
     * @param string $name 病人姓名
     * @param int $age 病人年龄
     * @param string $gender 病人性别
     */
    public function assignPatient($name, $age, $gender){
        // 使用正确初始化的 Patient 对象添加到病人列表
        $this->patients[] = new Patient($name, $age, $gender);
    }

    /**
     * 根据索引从诊所中删除一位病人。
     *
     * @param int $index 要删除病人的索引
     */
    public function deletePatient($index){
        if (isset($this->patients[$index])) {
            unset($this->patients[$index]);
            // 重置数组索引以避免空洞,可选操作
            $this->patients = array_values($this->patients);
        }
    }
}
?>

4. 完整示例代码与运行结果

结合上述对 Patient 类构造函数的修正和 Clinic 类聚合关系的调整,以下是完整的优化代码示例:

name = $name;
        $this->age = $age;
        $this->gender = $gender;
    }

    public function getName(){
        return $this->name;
    }

    public function getAge(){
        return $this->age;
    }

    public function getGender(){
        return $this->gender;
    }
}

// Clinic 类定义
class Clinic {
    private $patients = [];

    public function getPatients(){
        return $this->patients;
    }

    public function assignPatient($name, $age, $gender){
        $this->patients[] = new Patient($name, $age, $gender);
    }

    public function deletePatient($index){
        if (isset($this->patients[$index])) {
            unset($this->patients[$index]);
            // 可选:重新索引数组,使键连续
            $this->patients = array_values($this->patients);
        }
    }
}

// 实例化并操作
$clinic = new Clinic();

$clinic->assignPatient("Patrick star", 18, "Male");
$clinic->assignPatient("SpongeBob Squarepants", 17, "Male");
$clinic->assignPatient("Eugene Krab", 28, "Male");

echo "Initial patient list:\n";
print_r($clinic->getPatients());

$clinic->deletePatient(1); // 删除索引为1的病人 (SpongeBob Squarepants)

echo "\nPatient list after deletion:\n";
print_r($clinic->getPatients());

?>

预期输出:

Initial patient list:
Array
(
    [0] => Patient Object
        (
            [name:Patient:private] => Patrick star
            [age:Patient:private] => 18
            [gender:Patient:private] => Male
        )

    [1] => Patient Object
        (
            [name:Patient:private] => SpongeBob Squarepants
            [age:Patient:private] => 17
            [gender:Patient:private] => Male
        )

    [2] => Patient Object
        (
            [name:Patient:private] => Eugene Krab
            [age:Patient:private] => 28
            [gender:Patient:private] => Male
        )

)

Patient list after deletion:
Array
(
    [0] => Patient Object
        (
            [name:Patient:private] => Patrick star
            [age:Patient:private] => 18
            [gender:Patient:private] => Male
        )

    [1] => Patient Object
        (
            [name:Patient:private] => Eugene Krab
            [age:Patient:private] => 28
            [gender:Patient:private] => Male
        )

)

从输出中可以看出,Patient 对象现在被正确初始化,并且 Clinic 类能够有效地管理病人列表,不再出现 NULL 值。

5. 注意事项与最佳实践

  • 始终使用 __construct 初始化对象: 养成在需要初始化属性的类中定义 __construct 方法的习惯,确保对象在创建时处于有效状态。
  • 仔细评估类之间的关系: 在设计类时,务必思考它们之间的真实关系。是“is-a”(继承)还是“has-a”(聚合/组合)?错误的类关系会导致设计缺陷、代码难以维护和理解。
  • 继承的滥用: 继承是一种强耦合关系,应谨慎使用。只有当子类确实是父类的一个特化版本时才考虑继承。过度使用继承可能导致“脆弱的基类问题”和复杂的类层次结构。
  • 聚合的灵活性: 聚合(或更强的组合)提供了更大的灵活性,允许在运行时动态地添加或移除对象,而不会强制建立父子关系。
  • 封装原则: 保持属性的私有性(private 或 protected),并通过公共的 getter/setter 方法来访问和修改它们,这是良好的封装实践。
  • 代码可读性与维护性: 清晰、逻辑合理的类设计能够显著提高代码的可读性和未来的可维护性。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

236

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

438

2024.03.01

go语言 面向对象
go语言 面向对象

本专题整合了go语言面向对象相关内容,阅读专题下面的文章了解更多详细内容。

56

2025.09.05

java面向对象
java面向对象

本专题整合了java面向对象相关内容,阅读专题下面的文章了解更多详细内容。

52

2025.11.27

俄罗斯Yandex引擎入口
俄罗斯Yandex引擎入口

2026年俄罗斯Yandex搜索引擎最新入口汇总,涵盖免登录、多语言支持、无广告视频播放及本地化服务等核心功能。阅读专题下面的文章了解更多详细内容。

143

2026.01.28

包子漫画在线官方入口大全
包子漫画在线官方入口大全

本合集汇总了包子漫画2026最新官方在线观看入口,涵盖备用域名、正版无广告链接及多端适配地址,助你畅享12700+高清漫画资源。阅读专题下面的文章了解更多详细内容。

28

2026.01.28

ao3中文版官网地址大全
ao3中文版官网地址大全

AO3最新中文版官网入口合集,汇总2026年主站及国内优化镜像链接,支持简体中文界面、无广告阅读与多设备同步。阅读专题下面的文章了解更多详细内容。

64

2026.01.28

php怎么写接口教程
php怎么写接口教程

本合集涵盖PHP接口开发基础、RESTful API设计、数据交互与安全处理等实用教程,助你快速掌握PHP接口编写技巧。阅读专题下面的文章了解更多详细内容。

2

2026.01.28

php中文乱码如何解决
php中文乱码如何解决

本文整理了php中文乱码如何解决及解决方法,阅读节专题下面的文章了解更多详细内容。

4

2026.01.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PHP课程
PHP课程

共137课时 | 9.9万人学习

JavaScript ES5基础线上课程教学
JavaScript ES5基础线上课程教学

共6课时 | 11.2万人学习

PHP新手语法线上课程教学
PHP新手语法线上课程教学

共13课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号