在C和C++里,父類都不允許調用子類的方法,但在php里可以。下面是一個調用的例子:
<?php abstract class Animal { protected $name; public function run() { echo 'Aniaml run'; $this->swim();
echo $this->fishNum;
} } class Fish extends Animal {
public $fishNum=10; public function swim() { echo 'Fish swim'; } } $fish=new Fish(); $fish->swim(); $fish->run();
輸出:
Fish swim
Aniaml run
Fish swim
10
我們在父類的run()方法里面調用了
$this->swim();swim是子類定義的方法(注意,我們並沒有在父類中定義abstract function swim()),說明了這不是多態。
我們不僅調用了方法,還調用了成員,是ok的。
一篇文章:
PHP父類調用子類方法,CodeIgniter中DB的繼承關系
先看幾行代碼:
class A { private $b; function __construct($a){ $this->b = $a; } function func1(){ var_dump($this->b); } function func2(){ $this->funcb(); } } class B extends A { function funcb(){ var_dump("b"); } } $a = new B('a'); $a->func1(); $a->func2();
這幾行代碼是沒有錯誤的,在父類中調用子類的方法,子類實例化之后可以正常工作。 CI的DB部分正是使用了這種方式來封裝數據庫操作。
CI的DB函數的寫法正是先加載CI_DB_driver
這個基類,然后檢查active record是否開啟,如果開啟則
1 |
class CI_DB_active_record extends CI_DB_driver |
然后
1 |
class CI_DB extends CI_DB_active_record |
否則
1 |
class CI_DB extends CI_DB_driver |
最后在require相應數據庫的driver,如mysql,
1 |
CI_DB_mysql_driver extends CI_DB |
在這幾個類中,只有CI_DB_driver
有構造函數且帶一個參數;
DB函數中這樣寫的:
1 |
$driver = 'CI_DB_' . $params [ 'dbdriver' ]. '_driver' ; |
2 |
$DB = new $driver ( $params ); |
$params正是傳給CI_DB_driver
的構造函數;
然后檢查是否自動初始化:如果是
1 |
$DB ->initialize(); |
而initialize方法位於CI_DB_driver
類中,此方法調用了子類的方法:
1 |
$this ->conn_id = ( $this ->pconnect == FALSE) ? $this ->db_connect() : $this ->db_pconnect(); |
db_connect
方法和db_pconnect
方法是在相映數據庫driver實現的。
如mysql的實現位於CI_DB_mysql_driver
類中:
function db_connect() { if ($this->port != '') { $this->hostname .= ':'.$this->port; } return @mysql_connect($this->hostname, $this->username, $this->password, TRUE); }