hi
又是開森的周六了。積攢的兩周的衣服,終於是差不多洗完了。大下午的才來學點東西~~
1、PHP面向對象(三)
四、OOP的高級實踐
4.3 Static-靜態成員
<?php
date_default_timezone_set("PRC");
/**
* 1. 類的定義以class關鍵字開始,後面跟著這個類的名稱。類的名稱命名通常每個單詞的第一個字母大寫。
* 2. 定義類的屬性
* 3. 定義類的方法
* 4. 實例化類的對象
* 5. 使用對象的屬性和方法
*/
class Human{
public $name;
public $height;
public $weight;
public function eat($food){
echo $this->name."'s eating ".$food."<br/>";
}
}
class Animal{
public $kind;
public $gender;
}
class NbaPlayer extends Human{
// 類的屬性的定義
public $name="Jordan";// 定義屬性
public $height="198cm";
public $weight="98kg";
public $team="Bull";
public $playerNumber="23";
private $age="44";
public $president="David Stern";
// 類的方法的定義
public function changePresident($newP){
$this->president=$newP;
}
public function run() {
echo "Running<br/>";
}
public function jump(){
echo "Jumping<br/>";
}
public function dribble(){
echo "Dribbling<br/>";
}
public function shoot(){
echo "Shooting<br/>";
}
public function dunk(){
echo "Dunking<br/>";
}
public function pass(){
echo "Passing<br/>";
}
public function getAge(){
echo $this->name."'s age is ".$this->age;
}
function __construct($name, $height, $weight, $team, $playerNumber){
print $name . ";" . $height . ";" . $weight . ";" . $team . ";" . $playerNumber."\n";
$this->name = $name; // $this是php裡面的偽變量,表示對象自身
$this->height = $height; // 通過$this可以設置對象的屬性值
$this->weight = $weight;
$this->team = $team;
$this->playerNumber = $playerNumber;
}
}
/**
* 1. 類實例化為對象時使用new關鍵字,new之後緊跟類的名稱和一對括號。
* 2. 使用對象可以像使用其他值一樣進行賦值操作
*/
$jordan = new NbaPlayer("Jordan", "198cm","98kg","Bull","23");echo "<br/>";
$james=new NbaPlayer("James", "203cm", "120kg", "Heat", "6");echo "<br/>";
// 訪問對象的屬性使用的語法是->符號,後面跟著屬性的名稱
echo $jordan->name."<br/>";
// 調用對象的某個方法使用的語法是->符號,後面跟著方法的名稱和一對括號
$jordan->run();
$jordan->pass();
//子類調用父類的方法
$jordan->eat("apple");
//試著調用private,直接以及通過內部的public函數
//$jordan->age;
$jordan->getAge();echo "<br/>";
$jordan->changePresident("Adam Silver");
echo $jordan->president."<br/>";
echo $james->president."<br/>";
直接從上述例子開始吧。
這裡想得到的是,把兩位對象的某一個變量同時變掉。——用static
public static $president="David Stern";
// 類的方法的定義
public static function changePresident($newP){
static::$president=$newP;//這裡static換成self更符合規范
}
注意這裡static的位置,以及,方法內的::
調用的方法也有所變動。
echo NbaPlayer::$president;echo "<br/>";
NbaPlayer::changePresident("Adam Silver");
echo NbaPlayer::$president;echo "<br/>";
也就是像之前說的,靜態成員就是個常量,所以不針對某個具體的對象(不受具體對象的約束)——基於此,定義&賦值&調用都不需要具體的對象參與。
內部調用要用self/static::$...
外部調用,類名::
用處就是所有的對象共用的數據。
--如果內部調用時,變量在父類中
比如說,在上述例子中,父類human中寫這麼一句話
public static $aaa="dafdfa";
然後在子類nbaplayer中,調用父類的靜態成員時,要
echo parent::$aaa;
而外部調用,按照上面說的,類名::,所以,直接父類類名即可
echo Human::$aaa;
--其他
在靜態方法中,是無法訪問其他變量的,也就是說,不能用$this->
--小結
/**
* 靜態成員
* 1. 靜態屬性用於保存類的公有數據
* 2. 靜態方法裡面只能訪問靜態屬性
* 3. 靜態成員不需要實例化對象就可以訪問
* 4. 類內部,可以通過self或者static關鍵字訪問自身的靜態成員
* 5. 可以通過parent關鍵字訪問父類的靜態成員
* 6. 可以通過類名稱在外部訪問類的靜態成員
*/
4.4 Final成員
--問題
不希望某個類擁有子類;
不希望子類修改父類中的某個變量(避免重寫?)
--final
》=php5版本
舉個例子
class BaseClass{
public function test(){
echo "BaseClass::test called<br/>";
}
public function test1(){
echo "BaseClass::test1 called<br/>";
}
}
class ChildClass extends BaseClass{
public function test(){
echo "ChildClass::test called<br/>";
}
}
$obj=new ChildClass();
$obj->test();
子類中編寫跟父類中完全一致的方法名(內容可以不同),會完成對父類方法的重寫!
所以,不希望被重寫的父類中的方法,寫上final
final public function test(){
依此類推,對於不想有子類的父類,在類名那裡寫上final
final class BaseClass{
--小結
/**
* 重寫和Final
* 1. 子類中編寫跟父類完全一致的方法可以完成對父類方法的重寫
* 2. 對於不想被任何類繼承的類可以在class之前添加final關鍵字
* 3. 對於不想被子類重寫(overwrite, 修改)的方法,可以在方法定義前面添加final關鍵字
*/
4.5 數據訪問
先把final都去掉
--parent
然後再子類中的方法中寫
parent::test();
運行後會發現,依然可以通過parent的關鍵字調用父類中,即使是被重寫的數據
--self
然後在父類中的方法test中寫
self::test1();
運行後,發現self可以調用同一個類中的數據(其他方法/靜態變量/常量const)
--小結
/**
* 數據訪問補充
* 1. parent關鍵字可以用於調用父類被重寫的類成員
* 2. self關鍵字可以用於訪問類自身的成員方法,也可以用於訪問自身的靜態成員和類常量;不能用於訪問類自身的屬性;訪問類常量時不用在常量名稱前面加$符號
* 3. static關鍵字用於訪問類自身定義的靜態成員,訪問靜態屬性時需要在屬性名前面添加$符號
*/
4.6 對象接口
非常重要!!!
--問題
不同的類,有著相同的行為,但相同的行為又有著不同的實現方法。
比如人和動物都會吃東西,但吃的方式方法又不太一樣。
--定義
接口就是把不同類的共同行為進行了定義,然後在不同的類裡面實現不同的功能。
--栗子
//定義一個接口
interface ICanEat{
public function eat($food);
}
可以看到,接口中並沒有方法的具體實現,但必須有方法!
那麼,下面是,“人類會吃”
//具體對象,連接到接口
class Human implements ICanEat{
public function eat($food){
echo "Human eating ".$food.".<br/>";
}
}
$obj=new Human();
$obj->eat("shit");
請忽略我給出的“食物”。
注意,不再用extends,而是implements。然後,同樣是方法名的完全相同。然後,對象必須/最好實現方法。
繼續
interface ICanEat{
public function eat($food);
}
//具體對象,連接到接口
class Human implements ICanEat{
public function eat($food){
echo "Human eating ".$food.".<br/>";
}
}
class Animal implements ICanEat{
public function eat($food){
echo "Animal eating ".$food.".<br/>";
}
}
$obj=new Human();
$obj->eat("shit");
$monkey=new Animal();
$monkey->eat("banana");
讓動物也吃起來!
--逆向操作
判斷某個對象是否連接了某個接口。
var_dump($obj instanceof ICanEat);
會返回boolean值。
--更多的栗子
interface ICanPee extends ICanEat{
public function pee();
}
class Demon implements ICanPee{
public function pee(){
echo "Can demon pee?";
}
public function eat($food){
echo "Can demon eat ".$food;
}
}
$ghost=new Demon();
$ghost->pee();
$ghost->eat("shit");
接口本質上也是類,可以被繼承/繼承,但是使用繼承接口的接口,所有父類、“爺類”的方法都要有具體實現。
--小結
/**
* 接口
* 1. 接口的基本概念和基本使用方法
* 2. 接口裡面的方法沒有具體的實現
* 3. 實現了某個接口的類必須提供接口中定義的方法
* 4. 不能用接口創建對象,但是能夠判斷某個對象是否實現了某個接口
* 5. 接口可以繼承接口(interface extends interface)
* 6. 接口中定義的所有方法都必須是公有,這是接口的特性。
*/
aaaaaaaaaaaaaa
bu xiang xie le....................
ming tian yao ge ..............