在面向對象的程序設計(英語:Object-oriented programming,縮寫:OOP)中,對象是一個由信息及對信息進行處理的描述所組成的總體,是對現實世界的抽象。
在現實世界裏咱們所面對的事情都是對象,如計算機、電視機、自行車等。
對象的主要三個特性:
對象的行爲:能夠對 對象施加那些操做,開燈,關燈就是行爲。
對象的形態:當施加那些方法是對象如何響應,顏色,尺寸,外型。
對象的表示:對象的表示就至關於身份證,具體區分在相同的行爲與狀態下有什麼不一樣。
好比 Animal(動物) 是一個抽象類,咱們能夠具體到一隻狗跟一隻羊,而狗跟羊就是具體的對象,他們有顏色屬性,能夠寫,能夠跑等行爲狀態。
1020.png
面向對象內容
類 − 定義了一件事物的抽象特色。類的定義包含了數據的形式以及對數據的操做。
對象 − 是類的實例。
成員變量 − 定義在類內部的變量。該變量的融金匯銀值對外是不可見的,可是能夠經過成員函數訪問,在類被實例化爲對象後,該變量便可稱爲對象的屬性。
成員函數 − 定義在類的內部,可用於訪問對象的數據。
繼承 − 繼承性是子類自動共享父類數據結構和方法的機制,這是類之間的一種關係。在定義和實現一個類的時候,能夠在一個已經存在的類的基礎之上來進行,把這個已經存在的類所定義的內容做爲本身的內容,並加入若干新的內容。
父類 − 一個類被其餘類繼承,可將該類稱爲父類,或基類,或超類。
子類 − 一個類繼承其餘類稱爲子類,也可稱爲派生類。
多態 − 多態性是指相同的操做或函數、過程可做用於多種類型的對象上並得到不一樣的結果。不一樣的對象,收到同一消息能夠產生不一樣的結果,這種現象稱爲多態性。
重載 − 簡單說,就是函數或者方法有一樣的名稱,可是參數列表不相同的情形,這樣的同名不一樣參數的函數或者方法之間,互相稱之爲重載函數或者方法。
抽象性 − 抽象性是指將具備一致的數據結構(屬性)和行爲(操做)的對象抽象成類。一個類就是這樣一種抽象,它反映了與應用有關的重要性質,而忽略其餘一些無關內容。任何類的劃分都是主觀的,但必須與具體的應用有關。
封裝 − 封裝是指將現實世界中存在的某個客體的屬性與行爲綁定在一塊兒,並放置在一個邏輯單元內。
構造函數 − 主要用來在建立對象時初始化對象, 即爲對象成員變量賦初始值,總與new運算符一塊兒使用在建立對象的語句中。
析構函數 − 析構函數(destructor) 與構造函數相反,當北京股票配資公司對象結束其生命週期時(例如對象所在的函數已調用完畢),系統自動執行析構函數。析構函數每每用來作"清理善後" 的工做(例如在創建對象時用new開闢了一片內存空間,應在退出前在析構函數中用delete釋放)。
下圖中咱們經過 Car 類 建立了三個對象:Mercedes, Bmw, 和 Audi。php
$mercedes = new Car ();
$bmw = new Car ();
$audi = new Car ();數據結構
1012.jpg
PHP 類定義
PHP 定義類一般語法格式以下:
<?php
class phpClass {
var $var1;
var $var2 = "constant string";
function myfunc ($arg1, $arg2) {
[..]
}
[..]
}
?>
解析以下:
類使用 class 關鍵字後加上類名定義。
類名後的一對大括號({})內能夠定義變量和方法。
類的變量使用 var 來聲明, 變量也能夠初始化值。
函數定義相似 PHP 函數的定義,但函數只能經過該類及其實例化的對象訪問。
實例
<?php
class Site {
/* 成員變量 */
var $url;
var $title;
/* 成員函數 */
function setUrl($par){
$this->url = $par;
}
function getUrl(){
echo $this->url . PHP_EOL;
}
function setTitle($par){
$this->title = $par;
}
function getTitle(){
echo $this->title . PHP_EOL;
}
}
?>
變量 $this 表明自身的對象。
PHP_EOL 爲換行符。
PHP 中建立對象
類建立後,咱們可使用 new 運算符來實例化該類的對象:
$php = new Site;
$taobao = new Site;
$google = new Site;
以上代碼咱們建立了三個對象,三個對象各自都是獨立的,接下來咱們來看看如何訪問成員方法與成員變量。
調用成員方法
在實例化對象後,咱們可使用該對象調用成員方法,該對象的北京股票配資公司成員方法只能操做該對象的成員變量:
// 調用成員函數,設置標題和URL
$php->setTitle( "php中文網" );
$taobao->setTitle( "淘寶" );
$google->setTitle( "Google 搜索" );ide
$php->setUrl( 'www.php.cn' );
$taobao->setUrl( 'www.taobao.com' );
$google->setUrl( 'www.google.com' );函數
// 調用成員函數,獲取標題和URL
$php->getTitle();
$taobao->getTitle();
$google->getTitle();ui
$php->getUrl();
$taobao->getUrl();
$google->getUrl();
完整代碼以下:
實例
<?php
class Site {
/* 成員變量 */
var $url;
var $title;
/* 成員函數 */
function setUrl($par){
$this->url = $par;
}
function getUrl(bjrongjinhuiyin.com){
echo $this->url . PHP_EOL;
}
function setTitle($par){
$this->title = $par;
}
function getTitle(){
echo $this->title . PHP_EOL;
}
} this
$php = new Site;
$taobao = new Site;
$google = new Site; google
// 調用成員函數,設置標題和URL
$php->setTitle( "php中文網" );
$taobao->setTitle( "淘寶" );
$google->setTitle( "Google 搜索" ); url
$php->setUrl( 'www.php.cn' );
$taobao->setUrl( 'www.taobao.com' );
$google->setUrl( 'www.google.com' ); 設計
// 調用成員函數,獲取標題和URL
$php->getTitle();
$taobao->getTitle();
$google->getTitle(); 對象
$php->getUrl();
$taobao->getUrl();
$google->getUrl();
?>
運行實例 »
點擊 "運行實例" 按鈕查看在線實例
執行以上代碼,輸出結果爲:
php中文網
淘寶
Google 搜索
www.php.cn
www.taobao.com
www.google.com
PHP 構造函數
構造函數 ,是一種特殊的方法。主要用來在建立對象時初始化對象, 即爲對象成員變量賦初始值,總與new運算符一塊兒使用在建立對象的語句中。
PHP 5 允行開發者在一個類中定義一個方法做爲構造函數,語法格式以下:
void __construct ([ mixed $args [, $... ]] )
在上面的例子中咱們就能夠經過構造方法來初始化 $url 和 $title 變量:
function __construct( $par1, $par2 ) {
$this->url = $par1;
$this->title = $par2;
}
如今咱們就不須要再調用 setTitle 和 setUrl 方法了:
實例
<?php
class Site {
/* 成員變量 */
var $url;
var $title;
function __construct( $par1, $par2 ) {
$this->url = $par1;
$this->title = $par2;
}
/* 成員函數 */
function setUrl($par){
$this->url = $par;
}
function getUrl(){
echo $this->url . PHP_EOL;
}
function setTitle($par){
$this->title = $par;
}
function getTitle(){
echo $this->title . PHP_EOL;
}
}
$php = new Site('www.php.cn', 'php中文網');
$taobao = new Site('www.taobao.com', '淘寶');
$google = new Site('www.google.com', 'Google 搜索');
// 調用成員函數,獲取標題和URL
$php->getTitle();
$taobao->getTitle();
$google->getTitle();
$php->getUrl();
$taobao->getUrl();
$google->getUrl();
?>
運行實例 »
點擊 "運行實例" 按鈕查看在線實例
析構函數
析構函數(destructor) 與構造函數相反,當對象結束其生命週期時(例如對象所在的函數已調用完畢),系統自動執行析構函數。
PHP 5 引入了析構函數的概念,這相似於其它面向對象的語言,其語法格式以下:
void __destruct ( void )
實例
<?php
class MyDestructableClass {
function __construct() {
print "構造函數\n";
$this->name = "MyDestructableClass";
}
function __destruct() {
print "銷燬 " . $this->name . "\n";
}
}
$obj = new MyDestructableClass();
?>
執行以上代碼,輸出結果爲:
構造函數
銷燬 MyDestructableClass
繼承
PHP 使用關鍵字 extends 來繼承一個類,PHP 不支持多繼承,格式以下:
class Child extends Parent {
// 代碼部分
}
實例
實例中 Child_Site 類繼承了 Site 類,並擴展了功能:
<?php
// 子類擴展站點類別
class Child_Site extends Site {
var $category;
function setCate($par){
$this->category = $par;
}
function getCate(){
echo $this->category . PHP_EOL;
}
}
方法重寫
若是從父類繼承的方法不能知足子類的需求,能夠對其進行改寫,這個過程叫方法的覆蓋(override),也稱爲方法的重寫。
實例中重寫了 getUrl 與 getTitle 方法:
function getUrl() {
echo $this->url . PHP_EOL;
return $this->url;
}
function getTitle(){
echo $this->title . PHP_EOL;
return $this->title;
}
訪問控制
PHP 對屬性或方法的訪問控制,是經過在前面添加關鍵字 public(公有),protected(受保護)或 private(私有)來實現的。
public(公有):公有的類成員能夠在任何地方被訪問。
protected(受保護):受保護的類成員則能夠被其自身以及其子類和父類訪問。
private(私有):私有的類成員則只能被其定義所在的類訪問。
屬性的訪問控制
類屬性必須定義爲公有,受保護,私有之一。若是用 var 定義,則被視爲公有。
<?php
/**
* Define MyClass
*/
class MyClass
{
public $public = 'Public';
protected $protected = 'Protected';
private $private = 'Private';
function printHello()
{
echo $this->public;
echo $this->protected;
echo $this->private;
}
}
$obj = new MyClass();
echo $obj->public; // 這行能被正常執行
echo $obj->protected; // 這行會產生一個致命錯誤
echo $obj->private; // 這行也會產生一個致命錯誤
$obj->printHello(); // 輸出 Public、Protected 和 Private
/**
* Define MyClass2
*/
class MyClass2 extends MyClass
{
// 能夠對 public 和 protected 進行重定義,但 private 而不能
protected $protected = 'Protected2';
function printHello()
{
echo $this->public;
echo $this->protected;
echo $this->private;
}
}
$obj2 = new MyClass2();
echo $obj2->public; // 這行能被正常執行
echo $obj2->private; // 未定義 private
echo $obj2->protected; // 這行會產生一個致命錯誤
$obj2->printHello(); // 輸出 Public、Protected2 和 Undefined
?>
方法的訪問控制
類中的方法能夠被定義爲公有,私有或受保護。若是沒有設置這些關鍵字,則該方法默認爲公有。
<?php
/**
* Define MyClass
*/
class MyClass
{
// 聲明一個公有的構造函數
public function __construct() { }
// 聲明一個公有的方法
public function MyPublic() { }
// 聲明一個受保護的方法
protected function MyProtected() { }
// 聲明一個私有的方法
private function MyPrivate() { }
// 此方法爲公有
function Foo()
{
$this->MyPublic();
$this->MyProtected();
$this->MyPrivate();
}
}
$myclass = new MyClass;
$myclass->MyPublic(); // 這行能被正常執行
$myclass->MyProtected(); // 這行會產生一個致命錯誤
$myclass->MyPrivate(); // 這行會產生一個致命錯誤
$myclass->Foo(); // 公有,受保護,私有均可以執行
/**
* Define MyClass2
*/
class MyClass2 extends MyClass
{
// 此方法爲公有
function Foo2()
{
$this->MyPublic();
$this->MyProtected();
$this->MyPrivate(); // 這行會產生一個致命錯誤
}
}
$myclass2 = new MyClass2;
$myclass2->MyPublic(); // 這行能被正常執行
$myclass2->Foo2(); // 公有的和受保護的均可執行,但私有的不行
class Bar
{
public function test() {
$this->testPrivate();
$this->testPublic();
}
public function testPublic() {
echo "Bar::testPublic\n";
}
private function testPrivate() {
echo "Bar::testPrivate\n";
}
}
class Foo extends Bar
{
public function testPublic() {
echo "Foo::testPublic\n";
}
private function testPrivate() {
echo "Foo::testPrivate\n";
}
}
$myFoo = new foo();
$myFoo->test(); // Bar::testPrivate
// Foo::testPublic
?>
接口
使用接口(interface),能夠指定某個類必須實現哪些方法,但不須要定義這些方法的具體內容。
接口是經過 interface 關鍵字來定義的,就像定義一個標準的類同樣,但其中定義全部的方法都是空的。
接口中定義的全部方法都必須是公有,這是接口的特性。
要實現一個接口,使用 implements 操做符。類中必須實現接口中定義的全部方法,不然會報一個致命錯誤。類能夠實現多個接口,用逗號來分隔多個接口的名稱。
<?php
// 聲明一個'iTemplate'接口
interface iTemplate
{
public function setVariable($name, $var);
public function getHtml($template);
}
// 實現接口
class Template implements iTemplate
{
private $vars = array();
public function setVariable($name, $var)
{
$this->vars[$name] = $var;
}
public function getHtml($template)
{
foreach($this->vars as $name => $value) {
$template = str_replace('{' . $name . '}', $value, $template);
}
return $template;
}
}
常量
能夠把在類中始終保持不變的值定義爲常量。在定義和使用常量的時候不須要使用 $ 符號。
常量的值必須是一個定值,不能是變量,類屬性,數學運算的結果或函數調用。
自 PHP 5.3.0 起,能夠用一個變量來動態調用類。但該變量的值不能爲關鍵字(如 self,parent 或 static)。
實例
<?php
class MyClass
{
const constant = '常量值';
function showConstant() {
echo self::constant . PHP_EOL;
}
}
echo MyClass::constant . PHP_EOL;
$classname = "MyClass";
echo $classname::constant . PHP_EOL; // 自 5.3.0 起
$class = new MyClass();
$class->showConstant();
echo $class::constant . PHP_EOL; // 自 PHP 5.3.0 起
?>
抽象類
任何一個類,若是它裏面至少有一個方法是被聲明爲抽象的,那麼這個類就必須被聲明爲抽象的。
定義爲抽象的類不能被實例化。
被定義爲抽象的方法只是聲明瞭其調用方式(參數),不能定義其具體的功能實現。
繼承一個抽象類的時候,子類必須定義父類中的全部抽象方法;另外,這些方法的訪問控制必須和父類中同樣(或者更爲寬鬆)。例如某個抽象方法被聲明爲受保護的,那麼子類中實現的方法就應該聲明爲受保護的或者公有的,而不能定義爲私有的。此外方法的調用方式必須匹配,即類型和所需參數數量必須一致。例如,子類定義了一個可選參數,而父類抽象方法的聲明裏沒有,則二者的聲明並沒有衝突。
<?php
abstract class AbstractClass
{
// 強制要求子類定義這些方法
abstract protected function getValue();
abstract protected function prefixValue($prefix);
// 普通方法(非抽象方法)
public function printOut() {
print $this->getValue() . PHP_EOL;
}
}
class ConcreteClass1 extends AbstractClass
{
protected function getValue() {
return "ConcreteClass1";
}
public function prefixValue($prefix) {
return "{$prefix}ConcreteClass1";
}
}
class ConcreteClass2 extends AbstractClass
{
public function getValue() {
return "ConcreteClass2";
}
public function prefixValue($prefix) {
return "{$prefix}ConcreteClass2";
}
}
$class1 = new ConcreteClass1;
$class1->printOut();
echo $class1->prefixValue('FOO_') . PHP_EOL;
$class2 = new ConcreteClass2;
$class2->printOut();
echo $class2->prefixValue('FOO_') . PHP_EOL;
?>
執行以上代碼,輸出結果爲:
ConcreteClass1
FOO_ConcreteClass1
ConcreteClass2
FOO_ConcreteClass2
Static 關鍵字
聲明類屬性或方法爲 static(靜態),就能夠不實例化類而直接訪問。
靜態屬性不能經過一個類已實例化的對象來訪問(但靜態方法能夠)。
因爲靜態方法不須要經過對象便可調用,因此僞變量 $this 在靜態方法中不可用。
靜態屬性不能夠由對象經過 -> 操做符來訪問。
自 PHP 5.3.0 起,能夠用一個變量來動態調用類。但該變量的值不能爲關鍵字 self,parent 或 static。
<?php
class Foo {
public static $my_static = 'foo';
public function staticValue() {
return self::$my_static;
}
}
print Foo::$my_static . PHP_EOL;
$foo = new Foo();
print $foo->staticValue() . PHP_EOL;
?>
執行以上程序,輸出結果爲:
foo
foo
Final 關鍵字
PHP 5 新增了一個 final 關鍵字。若是父類中的方法被聲明爲 final,則子類沒法覆蓋該方法。若是一個類被聲明爲 final,則不能被繼承。
如下代碼執行會報錯:
<?php
class BaseClass {
public function test() {
echo "BaseClass::test() called" . PHP_EOL;
}
final public function moreTesting() {
echo "BaseClass::moreTesting() called" . PHP_EOL;
}
}
class ChildClass extends BaseClass {
public function moreTesting() {
echo "ChildClass::moreTesting() called" . PHP_EOL;
}
}
// 報錯信息 Fatal error: Cannot override final method BaseClass::moreTesting()
?>
調用父類構造方法
PHP 不會在子類的構造方法中自動的調用父類的構造方法。要執行父類的構造方法,須要在子類的構造方法中調用 parent::__construct() 。
<?php
class BaseClass {
function __construct() {
print "BaseClass 類中構造方法" . PHP_EOL;
}
}
class SubClass extends BaseClass {
function __construct() {
parent::__construct(); // 子類構造方法不能自動調用父類的構造方法
print "SubClass 類中構造方法" . PHP_EOL;
}
}
class OtherSubClass extends BaseClass {
// 繼承 BaseClass 的構造方法
}
// 調用 BaseClass 構造方法
$obj = new BaseClass();
// 調用 BaseClass、SubClass 構造方法
$obj = new SubClass();
// 調用 BaseClass 構造方法$obj = new OtherSubClass();?>執行以上程序,輸出結果爲:BaseClass 類中構造方法BaseClass 類中構造方法SubClass 類中構造方法BaseClass 類中構造方法