freeCodeCamp/guide/chinese/php/class-inheritance/index.md

6.5 KiB
Raw Blame History

title localeTitle
Class Inheritance 类继承

类继承

在面向对象的程序设计中重新使用继承代码

在这里,我们将讨论如何通过使用继承来重用我们编写的代码而不需要任何代码重复。

人类

这是我们的Man类:

<?php 
 class Man 
 { 
    // 1. Declare the class variables 
    public $name; 
    protected $age; 
    public $height; 
    public $fav_sports; 
    private $fav_drinks; 
 
    // 2. Create a constructor method with 3 required parameters: name, age and height 
    public function __construct($name, $age, $height) 
    { 
        // 2A. Assign the values of parameters to class properties 
        // Also known as instance variables 
        // Using "$this->property_name" 
        $this->name = $name; 
        $this->age = $age; 
        $this->height = $height; 
 
        // 2B. Print out the man's attributes and values upon instantiation 
        echo "Our man's name is: " . $this->name . "\n"; 
        echo "He is " . $this->age . " years old and " . $this->height . " tall."; 
    } 
 
    // 3. Create class methods 
    public function giveFirmHandshakes() 
    { 
        return "I give firm handshakes."; 
    } 
 
    public function beStubborn() 
    { 
        return "I am stubborn."; 
    } 
 
    public function notPutToiletPaper() 
    { 
        return "It's not humanly possible to remember to put toilet paper rolls when they are finished"; 
    } 
 
    // 4. Age getter method 
    public function getAge() 
    { 
        return $this->age; 
    } 
 
    // Age setter method 
    public function setAge($age) 
    { 
        $this->age = $age; 
    } 
 
    // 5. Favorite Drinks setter method 
    public function setFavDrinks($drinks = array()) 
    { 
        if ($drinks) { 
            $this->fav_drinks = $drinks; 
        } 
    } 
 
    // Favorite Drinks getter method 
    public function getFavDrinks() 
    { 
        return $this->fav_drinks; 
    } 
 } 

健康的人

假设我们要创建另一个名为HealthyMan类,它具有Man类的所有属性和方法。

无需重新编写Man类的所有代码我们可以使用关键字extends来重用该代码。

<?php 
 class HealthyMan extends Man 
 { 
 
 } 

现在我们拥有来自HealthyMan里面的Man的所有类属性和方法。我们可以实例化HealthyMan类来快速检查这个。

<?php 
 $jackie = new HealthyMan('Jackie', 25, '5\' 5"'); 
 // => Our man's name is: Jackie 
 // => He is 25 years old and 5' 5" tall. 

我们可以继续设置HealthyMan又名Jackie最喜欢的运动和饮料。

<?php 
 $jackie->fav_sports = ['swimming', 'weight training']; 
 print_r($jackie->fav_sports); 
 // => 
 // Array 
 // ( 
 //     [0] => swimming 
 //     [1] => weight training 
 // ) 
 
 $jackie->setFavDrinks(['Matcha tea', 'Oolong Tea']); 
 print_r($jackie->getFavDrinks()); 
 // => 
 // Array 
 // ( 
 //     [0] => Matcha tea 
 //     [1] => Oolong Tea 
 // ) 

现在让我们看看我们是否可以调用Man的类方法giveFirmHandshakes() beStubborn()notPutToiletPaper()

<?php 
 echo "\n" . $jackie->giveFirmHandshakes(); 
 // => I give firm handshakes. 
 
 echo "\n" . $jackie->beStubborn(); 
 // => I am stubborn. 
 
 echo "\n" . $jackie->notPutToiletPaper(); 
 // => It's not humanly possible to remember to put toilet paper rolls when they are finished 

我们通过使用关键字extends继承Man类来获得所有这些。

一个真正健康的人

如果我们从Man类继承HealthyMan并且不做任何事情,那么它就胜过了整个目的。

HealthyMan类有类似的附加属性body_fat_percentageworkout_per_week ,以及类似的方法eatHealthy() meditateDaily()laughOften()

由于这些是个人属性我们可以设置它们对protected或private的可见性并为完全封装创建setter / getter方法。

<?php 
 class HealthyMan extends Man 
 { 
    /** 
     * HealthyMan properties 
     */ 
    private $body_fat_percentage; 
    private $workout_per_week; 
 
    /** 
     * HealthyMan methods 
     */ 
    public function eatHealthy() 
    { 
        return "I only eat healthy meals."; 
    } 
 
    public function meditateDaily() 
    { 
        return "I set aside 20 minutes daily to meditate."; 
    } 
 
    public function laughOften() 
    { 
        return "I watch funny TV shows to unwind myself."; 
    } 
 
    /** 
     * HealthyMan Setters and Getters 
     */ 
    public function setBodyFatPercentage($fat_percentage) 
    { 
        $this->body_fat_percentage  = $fat_percentage; 
    } 
 
    public function getBodyFatPercentage() 
    { 
        return $this->body_fat_percentage; 
    } 
 
    public function setWorkoutPerWeek($workout_times) 
    { 
        $this->workout_per_week = $workout_times; 
    } 
 
    public function getWorkoutPerWeek() 
    { 
        return $this->workout_per_week; 
    } 
 } 

我们可以调用这些方法来查看它们是否按预期工作:

<?php 
 
 echo "\n" . $jackie->eatHealthy(); 
 // => I only eat healthy meals. 
 
 echo "\n" . $jackie->meditateDaily(); 
 // => I set aside 20 minutes daily to meditate. 
 
 echo "\n" . $jackie->laughOften(); 
 // => I watch funny TV shows to unwind myself. 
 
 $jackie->setBodyFatPercentage(12); 
 echo "\nBody Fat %: " . $jackie->getBodyFatPercentage(); 
 // => Body Fat %: 12 
 
 $jackie->setWorkoutPerWeek(5); 
 echo "\nWorkout Times Per Week: " . $jackie->getWorkoutPerWeek(); 
 // => Workout Times Per Week: 5 

我们已成功重用现有代码并实现了子类。

他是顽固的吗?

虽然他继承了Man类的beStubborn() 因为Jackie是一个健康的男人他只是偶尔顽固一次。我们可以通过改变父类的方法让健康人的beStubborn()方法说“我偶尔会顽固”,而不仅仅是简单的“我很固执”。

<?php 
 class HealthyMan extends Man 
 { 
    ..... 
    ..... 
 
    public function beStubborn() 
    { 
        return "I am stubborn once in a while."; 
    } 
 
    ..... 
    ..... 
 } 

现在当我们可以使用Jackie的beStubborn()方法时,我们将看到与以前不同的输出:

<?php 
 echo "\n" . $jackie->beStubborn(); 
 // => I am stubborn once in a while. 

这演示了方法覆盖如何在OOP中工作。

通过使用方法重写,我们基本上在子类中重新声明父类的方法。

这样,父类的任何实例都维护其原始方法,而子类的任何实例都具有修改或重写的方法。