ThinkChat2.0新版上线,更智能更精彩,支持会话、画图、阅读、搜索等,送10W Token,即刻开启你的AI之旅 广告
在学习PHP 这种语言中你会发现, PHP中的方法是不能重载的, 所谓的方法重载就是定义相同的方法名,通过“参数的个数“不同或“参数的类型“不 同,来访问我们的相同方法名的不同方法。但是因为PHP是弱类型的语言, 所以在方法的参数中本身就可以接收不同类型的数据,又因为PHP的方法可以接收不定个数的参数,所以通过传递不同个数的参数调用不相同方法名的不同方法也是不成立的。所以在PHP里面没有方法重载。不能重载也就是在你的项目中不能定义相同方法名的方法。另外,因为PHP没有名子空间的概念,在同一个页面和被包含的页面中不能定义相同名称的方法, 也不能定义和PHP给我提供的方法的方法重名,当然在同一个类中也不能定义相同名称的方法。 我们这里所指的重载新的方法所指的是什么呢?其实我们所说的重载新的方法就是子类覆盖父类的已有的方法,那为什么要这么做呢?父类的方法不是可以继承过 来直接用吗?但有一些情况是我们必须要覆盖的,比如说我们前面提到过的例子里面, “Person”这个人类里面有一个“说话”的方法,所有继承“Person”类的子类都是可以“说话”的, 我们“Student”类就是“Person”类的子类,所以“Student”的实例就可以“说话“了, 但是人类里面“说话”的方法里面说出的是“Person”类里面的属性, 而“Student”类对“Person”类进行了扩展,又扩展出了几个新的属性,如果使用继承过来的“say()”说话方法的话,只能说出从 “Person”类继承过来的那些属性,那么新扩展的那些属性使用这个继承过来的“say()”的方法就说不出来了,那有的人就问了,我在 “Student”这个子类中再定义一个新的方法用于说话,说出子类里面所有的属性不就行了吗?一定不要这么做, 从抽象的角度来讲, 一个“学生”不能有两种“说话”的方法,就算你定义了两个不同的说话的方法,可以实现你想要的功能,被继承过来的那个“说话“方法可能没有机会用到了,而 且是继承过来的你也删不掉。这个时候我们就要用到覆盖了。 虽然说在PHP里面不能定义同名的方法, 但是在父子关系的两个类中,我们可以在子类中定义和父类同名的方法,这样就把父类中继承过来的方法覆盖掉了。 ~~~ 1 <? 2 //定义一个"人"类做为父类 3 class Person 4 { 5 //下面是人的成员属性 6 var $name; //人的名子 7 var $sex; //人的性别 8 var $age; //人的年龄 9 10 //定义一个构造方法参数为属性姓名$name、性别$sex和年龄$age进行赋值 11 function __construct($name, $sex, $age) { 12 $this->name = $name; 13 $this->sex = $sex; 14 $this->age = $age; 15 } 16 17 //这个人可以说话的方法, 说出自己的属性 18 function say() { 19 echo "我的名子叫:" . $this->name . " 性别:" . $this->sex . " 我的年龄是:" . $this->age; 20 } 21 } 22 23 class Student extends Person 24 { 25 26 var $school; //学生所在学校的属性 27 28 //这个学生学习的方法 29 function study() { 30 echo "我的名子叫:" . $this->name . " 我正在" . $this->school . " 学习"; 31 } 32 33 //这个学性可以说话的方法, 说出自己所有的属性,覆盖了父类的同名方法 34 function say() { 35 echo "我的名子叫:" . $this->name . " 性别:" . $this->sex . " 我的年龄是:" . $this->age . " 我在" . $this->school . "上学"; 36 } 37 } 38 ?> ~~~ 上面的例子, 我们就在“Student”子类里覆盖了继承父类里面的”say()”的方法,通过覆盖我们就实现了对“方法”扩展。但是,像这样 做虽然解决了我们上面说的问题,但是在实际开发中,一个方法不可能就一条代码或是几条代码,比如说“Person”类里面的“say()”方法有里面有 100条代码,如果我们想对这个方法覆盖保留原有的功能外加上一点点功能,就要把原有的100条代码重写一次, 再加上扩展的几条代码,这还算是好的,而有的情况,父类中的方法是看不见原代码的,这个时候你怎么去重写原有的代码呢?我们也有解决的办法,就是在子类这 个方法中可以调用到父类中被覆盖的方法, 也就是把被覆盖的方法原有的功能拿过来再加上自己的一点功能,可以通过两种方法实现在子类的方法中调用父类被覆盖的方法: 一种是使用父类的“类名::“来调用父类中被覆盖的方法; 一种是使用“parent::”的方试来调用父类中被覆盖的方法; ~~~ 1 class Student extends Person 2 { 3 var $school; //学生所在学校的属性 4 5 //这个学生学习的方法 6 function study() { 7 echo "我的名子叫:" . $this->name . " 我正在" . $this->school . "学习"; 8 } 9 10 //这个学性可以说话的方法, 说出自己所有的属性,覆盖了父类的同名方法 11 function say() { 12 13 //使用父类的"类名::"来调用父类中被覆盖的方法; 14 // Person::say(); 15 16 //或者使用"parent::"的方试来调用父类中被覆盖的方法; 17 parent::say(); 18 19 //加上一点自己的功能 20 echo "我的年龄是:" . $this->age . " 我在" . $this->school . "上学"; 21 } 22 } ~~~ 现在用两种方式都可以访问到父类中被覆盖的方法,我们选那种方式最好呢?用户可能会发现自己写的代码访问了父类的变量和函数。如果子类非常精炼或者父类非 常专业化的时候尤其是这样。 不要用代码中父类文字上的名字,应该用特殊的名字 parent,它指的就是子类在 extends 声明中所指的父类的名字。这样做可以避免在多个地方使用父类的名字。如果继承树在实现的过程中要修改,只要简单地修改类中 extends 声明的部分。 同样,构造方法在子类中如果没有声明的话,也可以使用父类中的构造方法,如果子类中重新定义了一个构造方法也会覆盖掉父类中的构造方法,如果想使用新的构造方法为所有属性赋值也可以用同样的方式。 ~~~ 1 class Student extends Person 2 { 3 var $school; //学生所在学校的属性 4 5 function __construct($name, $sex, $age, $school) { 6 //使用父类中的方法为原有的属性赋值 7 parent::__construct($name, $sex, $age); 8 $this->school = $school; 9 } 10 11 //这个学生学习的方法 12 function study() { 13 echo "我的名子叫:" . $this->name . " 我正在" . $this->school . " 学习"; 14 } 15 16 //这个人可以说话的方法, 说出自己的属性 17 function say() { 18 parent::say(); 19 //加上一点自己的功能 20 echo "我的年龄是:" . $this->age . " 我在" . $this->school . "上学"; 21 } 22 } ~~~