主頁 > 軟體設計 > 深入理解面向物件,花費時間自后一次整理,以后只更新里面的內容

深入理解面向物件,花費時間自后一次整理,以后只更新里面的內容

2021-09-29 11:47:10 軟體設計

前言
我相信很多同行去另一家公司之后都會吐槽原來的代碼寫的太菜,這其實很大一部分原因都來自公司要求“小步快跑,快速迭代”,不同的人重復造輪子,代碼質量參差不齊,各種膠水式的代碼遍布SVN和Git,大家不敢動原來的代碼,只能在最后的地方進行修修補補,所以導致維護困難,

一、面向物件的概念
物件:物件是一些變數和方法都聚集在一起的物體
類:
屬性:一個屬性是類中的一個內部變數,但從本質上講,屬性是指在類本身,而不是在這個類中任何方法宣告的變數;
參考:兩個不同的變數名字指向相同的內容;

特點:
1、封裝、通過修飾符改變屬性或者函式的訪問權限,達到保護的作用,

權限修飾符
private: 只能在自己的本類里才能被訪問到;
protected: 本類和子類能訪問;
public: 本類和子類,還有類外部都可以訪問到;

2、繼承:子類繼承父類的屬性和方法,再進一步拓展自己的屬性和方法;

1、子類繼承父類,子類有父類所有的屬性和方法,在子類里面能夠操作父類中不是private修飾的屬性或者方法;
2、子類的方法會覆寫父類的方法;
3、php不會在子類的構造方法中自動呼叫父類的構造方法,要執行父類的構造方法,需要在子類的構造方法中呼叫parent::__construct();
parent:__destruct();

3、多型:通過繼承復用實作的;同一個操作作用于不同的類的實體,將產生不同的執行結果,或者不同類的物件收到相同的訊息時,將得到不同的結果,

<?php
class animal{
    function can(){
        echo "this function weill be re-write in the children";
    }
}
class cat extends animal{
    function can(){
        echo "I can climb";
    }
}
class dog extends animal{
    function can(){
        echo "I can swim";
    }
}
function test($obj){
    $obj->can();
}
test(new cat());
test(new dog()); 同一個操作作用于不同類的實體,結果也不相同
?>

二、構造方法和析構方法
建構式:__construct() 主要用來在創建物件時初始化物件,即為物件成員變數賦值,在創建物件的陳述句中與new運算子一起使用;
解構式:__destruct() 在某個物件的所有參考都被洗掉或者當物件被顯示銷毀時執行;

如何銷毀物件
1、unset(),直接賦值為null或者其他值;
2、執行完最后一行代碼時進行自動銷毀;

$a = new People();
$a = $b = $c;
unset($a);
//此時unset($a)只是$a=null不再指向此物件,但其他變數$b,$c仍然指向此物件,所以該物件不能銷毀;

解構式即使在使用exit()終止腳本運行時也會被呼叫

php8對建構式的屬性有了進一步的提升:

簡單的數值物件:正常的流程:定義變數,建構式引數定義,給引數賦值;

class Person{
	public string $name;
	public int $age;
	public function __construct($name = '',$age = 0){
		$this->name = $name;
		$this->age = $age;
	}
}

提升后的:一次性完成了成員變數定義,引數定義以及賦值的三個程序

class Person{
	public function __construct(private string name = '',public int $age = 0 ){
		
	}
}

放在構造器提升引數里的屬性會同時復制為屬性和引數

三、防止重寫和被擴展 final
1、如果父類中的方法被宣告為final,則不允許被子類方法覆寫,如果一個類被宣告為final,則不能被繼承;

理解:final意思為最終的,那就是不能對她進行任何操作,

四、介面
1、介面可以指定某個類必須實作哪些方法,但不需要定義這些方法的具體內容;
2、介面通過interface關鍵字來定義,但其中定義的所有的方法都是空的
3、介面中定義的方法必須是公有的
4、實作一個介面,使用implements運算子
5、類中必須實作介面中定義的所有方法
6、類可以實作多個介面,用逗號來分隔多個介面的名稱

<?php
interface Fruit
{

    const MAX_WEIGHT = 5;   //此處不用宣告,就是一個靜態常量
    function setName($name);
    function getName();
}
//實作介面
class Apple implements Fruit
{

    private $name;
    function getName() {
        return $this->name;
    }
    function setName($_name) {
        $this->name = $_name;
    }
}

$apple = new Apple(); //創建物件
$apple->setName("蘋果");
echo "創建了一個" . $apple->getName();
echo "<br />";
echo "MAX_GRADE is " . Apple::MAX_WEIGHT;   //靜態常量

//實作多個介面
Class Template implements iTemplate,yTemplate{
	private $age = 19;
	public function setName(){
		return $this->age;
	}
	public function getName($){
		return 'age is '."$age";
	}
}

五、抽象類
1、任何一個類,里面至少有一個方法是被宣告為抽象的,則這個類就必須被宣告為抽象類;
2、定義為抽象的類不能被實體化;必須先繼承抽象來,再實體化子類;
3、被定義為抽象的方法只是宣告了其調動的方式,不能定義其具體的功能實作;
4、繼承一個抽象類的類,子類必須定義父類中的所有抽象方法;并且這些方法的訪問控制必須和父類一樣或者更為寬松;比如某個抽象方法被宣告為受保護的,那么子類中實作的方法就必須為受保護的或者公有的,不能定義為私有的;

abstract class AbstractClass{
    // 定義抽象方法
    abstract protected function getValue();
    // 普通方法
    public function printOut(){
        print $this->getValue()."";
    }
}

class ConcreteClass extends AbstractClass{
    protected function getValue(){
        return "abstract ";//抽象方法的實作
    }
}
$class1 = new ConcreteClass;
$class1->printOut();

六、traits代碼復用
1、php一直是單繼承,無法同時從兩個基類中繼承屬性和方法
2、trait不能實體化,通過在類中使用use關鍵字宣告要組合的trait類名;

trait Driver{
	public $carName = "DMW";
	public function driving(){
		echo "driving  {$this->carName}";
	}
}
 
class Person{
	public function age(){
		echo "I am 18 years old";
	}
}

class Student extends Person(){
	use Driver;
	public function study(){
		echo "Learn to drive ";
	}
}
$student = new Student();
$student->study();
$student->age();
$student->driving();

結果:
Learn to drive;
i am 18 years old;
driving BMW;

3、當方法或者屬性同名時,當前類中的方法會覆寫trait的方法,而trait的方法又覆寫了基類的方法;
4、組合多個trait,可用都好分割,use trait1,trait2;
5、insteadof是使用某個方法替代另一個,而as是給方法取一個別名;

<?php
trait Trait1 {
    public function hello() {
        echo "Trait1::hello \n";
    }
    public function hi() {
        echo "Trait1::hi \n";
    }
}
trait Trait2 {
    public function hello() {
        echo "Trait2::hello\n";
    }
    public function hi() {
        echo "Trait2::hi\n";
    }
}
class Class1 {
    use Trait1, Trait2 {
        Trait2::hello insteadof Trait1;
        Trait1::hi insteadof Trait2;
    }
}
class Class2 {
    use Trait1, Trait2 {
        Trait2::hello insteadof Trait1;
        Trait1::hi insteadof Trait2;
        Trait2::hi as hei;
        Trait1::hello as hehe;
    }
}
$Obj1 = new Class1();
$Obj1->hello();
$Obj1->hi();
echo "\n";
$Obj2 = new Class2();
$Obj2->hello();
$Obj2->hi();
$Obj2->hei();
$Obj2->hehe();

結果:
Trait2::hello
Trait1::hi


Trait2::hello
Trait1::hi
Trait2::hi
Trait1::hello
<?php
trait Hello {
    public function hello() {
        echo "hello,我是周伯通\n";
    }
}
class Class1 {
    use Hello {
        hello as protected;
    }
}
class Class2 {
    use Hello {
        Hello::hello as private hi;
    }
}
$Obj1 = new Class1();
$Obj1->hello(); # 報致命錯誤,因為hello方法被修改成受保護的
$Obj2 = new Class2();
$Obj2->hello(); # 輸出: hello,我是周伯通,因為原來的hello方法仍然是公共的
$Obj2->hi();  # 報致命錯誤,因為別名hi方法被修改成私有的

一篇比較好的文章有助于理解traits

七、靜態方法和屬性
1、宣告類屬性或方法為static,就可以不實體化類而直接訪問;
2、無論函式呼叫多少次,只初始化一次;

八、魔術方法
定義:某些情況下,會自動呼叫的方法,稱為魔術方法;所有的魔術方法,必須宣告為public

魔術方法的引數都不能通過參考傳遞,
魔術方法不能被宣告為static

__clone:克隆方法,當物件被克隆時,將會自動呼叫;

class Human{
    var $age = 22;
    public function __clone(){
        echo "有人克隆我,假冒";
    }
}
$a = new Human();
$b = clone $a;
echo $b;
結果:有人克隆我,假冒

拓展:深拷貝和淺拷貝
深拷貝:賦值時值被完全賦值,完全的copy,對其中一個做出改變,不會影響另一個
淺拷貝:賦值時,參考賦值,對其中的一個修改,會影響另一個
=賦值時,普通物件是深拷貝,但對物件來說,是淺拷貝,

$m =1;
$n = $m ;此時$n1
$n = 2;
echo $m ;只是值進行賦值,深拷貝 ,值為1

_______________________________
class Test{
	public $a = 1;
}
$m = new Test();
$n = $m;參考賦值
$m->a = 2;
echo $n->a ;結果為2.淺拷貝

__get();當我們呼叫一個權限上不允許呼叫的屬性或者不存的屬性時,會直接走__get()魔術方法取值;

class Human{
    var $age = 22;
    protected $name = "duanjiaqi";
    private $sex = "1";
    public function __get($p){
        echo '您是想訪問我的'.$p.'屬性';
    }
}
$a = new Human();
echo $a->age.PHP_EOL;//此時age的權限是可以訪問的,所以過為22;
echo $a->name.PHP_EOL;//如果沒有__get()魔術方法的話,在編輯器里name直接報錯,因為屬性權限為受保護的,類外部是訪問不到的;有__get()魔術方法的話,結果為:您是想訪問我的name屬性
echo $a->sex.PHP_EOL;
echo $a->no;//類內部沒有no屬性,沒有__get()魔術方法的話直接報錯;在__get()魔術方法存在的話,結果為:您是想訪問我的no屬性

__set():當為無權操作的屬性或者不存在的屬性賦值時,自動呼叫__set()魔術方法;

class Human{
    var $age = 22;
    protected $name = "duanjiaqi";
    private $sex = "1";
    public function __set($a,$b){
        echo "設定的".$a."屬性,";
        echo "值為".$b.".";
    }
}
$a = new Human();
$a->sex = 33;//sex屬性為private,類外部訪問不到,所以直接走__set()方法,為sex賦值;結果為:設定的sex屬性,值為33.

__isset():當對不可訪問或者不存在的屬性呼叫isset()或empty()時,isset()或被呼叫;

__unset():當對不可訪問或者不存在的屬性呼叫unset()時,__unset()會被呼叫

class Human{
    private $age = 22;
    public function __isset($name)
    {
        // TODO: Implement __isset() method.
        echo 'unset';
    }
}
$a = new Human();
isset($a->age);//如果$age的屬性為private或者protected時,則結果為unset,如果$age的屬性是public時,則結果為空;

__call():呼叫不可見或者不存在或者無權限的方法時,自動呼叫;

__callStatic(): 在靜態背景關系中呼叫一個不可訪問的方法時,__callStatic()會被呼叫,

class MethodTest 
{
    public function __call($name, $arguments) 
    {
        // 注意: $name 的值區分大小寫
        echo "Calling object method '$name' "
             . implode(', ', $arguments). "\n";
    }

    public static function __callStatic($name, $arguments) 
    {
        // 注意: $name 的值區分大小寫
        echo "Calling static method '$name' "
             . implode(', ', $arguments). "\n";
    }
}

$obj = new MethodTest;
$obj->runTest('in object context');

MethodTest::runTest('in static context');

結果:Calling object method 'runTest' in object context
Calling static method 'runTest' in static context
?>

__sleep()和__wakeup() serialize()與unserialize()
1、serialize()函式會檢查類中是否存在魔術方法__sleep(),如果存在,則該方法會先被呼叫,然后才執行序列化操作;unserialize()會檢查是否存在一個__wakeup()方法,如果存在,則會先呼叫__wakeup方法,預先準備物件需要的資源,
2、__sleep()常用語提交未提交的資料

__toString():方法用于一個類被當成字串時應怎樣回應

class TestClass{
	public $foo;
	public function __construct($foo){
		$this->foo = $foo;
	}
	public function __toString(){
		return $this->foo;
	}
}
$class = new TestClass('Hello');
echo $class;

結果為:Hello;

轉載請註明出處,本文鏈接:https://www.uj5u.com/ruanti/304007.html

標籤:其他

上一篇:ROS中CMakelists.txt中各個欄位的說明

下一篇:顯揚科技|基于三維機器視覺的拆碼垛系統

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 面試突擊第一季,第二季,第三季

    第一季必考 https://www.bilibili.com/video/BV1FE411y79Y?from=search&seid=15921726601957489746 第二季分布式 https://www.bilibili.com/video/BV13f4y127ee/?spm_id_fro ......

    uj5u.com 2020-09-10 05:35:24 more
  • 第三單元作業總結

    1.前言 這應該是本學期最后一次寫作業總結了吧。總體來說,對作業的節奏也差不多掌握了,作業做起來的效率也更高了。雖然和之前的作業一樣,作業中都要用到新的知識,但是相比之前,更加懂得了如何利用工具以及資料。雖然之間卡過殼,但總體而言,這幾次作業還算完成的比較好。 2.作業程序總結 相比前兩個單元,此單 ......

    uj5u.com 2020-09-10 05:35:41 more
  • 北航OO(2020)第四單元博客作業暨課程總結博客

    北航OO(2020)第四單元博客作業暨課程總結博客 本單元作業的架構設計 在本單元中,由于UML圖具有比較清晰的樹形結構,因此我對其中需要進行查詢操作的元素進行了包裝,在樹的父節點中存盤所有孩子的參考。考慮到性能問題,我采用了快取機制,一次查詢后盡可能快取已經遍歷過的資訊,以減少遍歷次數。 本單元我 ......

    uj5u.com 2020-09-10 05:35:48 more
  • BUAA_OO_第四單元

    一、UML決議器設計 ? 先看下題目:第四單元實作一個基于JDK 8帶有效性檢查的UML(Unified Modeling Language)類圖,順序圖,狀態圖分析器 MyUmlInteraction,實際上我們要建立一個有向圖模型,UML中的物件(元素)可能與同級元素連接,也可與低級元素相連形成 ......

    uj5u.com 2020-09-10 05:35:54 more
  • 6.1邏輯運算子

    邏輯運算子 1. && 短路與 運算式1 && 運算式2 01.運算式1為true并且運算式2也為true 整體回傳為true 02.運算式1為false,將不會執行運算式2 整體回傳為false 03.只要有一個運算式為false 整體回傳為false 2. || 短路或 運算式1 || 運算式2 ......

    uj5u.com 2020-09-10 05:35:56 more
  • BUAAOO 第四單元 & 課程總結

    1. 第四單元:StarUml檔案決議 本單元采用了圖模型決議UML。 UML檔案可以抽象為圖、子圖、邊的邏輯結構。 在實作中,圖的節點包括類、介面、屬性,子圖包括狀態圖、順序圖等。 采用了三次遍歷UML元素的方法建圖,第一遍遍歷建點,第二、三次遍歷設定屬性、連邊,實作圖物件的初始化。這里借鑒了一些 ......

    uj5u.com 2020-09-10 05:36:06 more
  • 談談我對C# 多型的理解

    面向物件三要素:封裝、繼承、多型。 封裝和繼承,這兩個比較好理解,但要理解多型的話,可就稍微有點難度了。今天,我們就來講講多型的理解。 我們應該經常會看到面試題目:請談談對多型的理解。 其實呢,多型非常簡單,就一句話:呼叫同一種方法產生了不同的結果。 具體實作方式有三種。 一、多載 多載很簡單。 p ......

    uj5u.com 2020-09-10 05:36:09 more
  • Python 資料驅動工具:DDT

    背景 python 的unittest 沒有自帶資料驅動功能。 所以如果使用unittest,同時又想使用資料驅動,那么就可以使用DDT來完成。 DDT是 “Data-Driven Tests”的縮寫。 資料:http://ddt.readthedocs.io/en/latest/ 使用方法 dd. ......

    uj5u.com 2020-09-10 05:36:13 more
  • Python里面的xlrd模塊詳解

    那我就一下面積個問題對xlrd模塊進行學習一下: 1.什么是xlrd模塊? 2.為什么使用xlrd模塊? 3.怎樣使用xlrd模塊? 1.什么是xlrd模塊? ?python操作excel主要用到xlrd和xlwt這兩個庫,即xlrd是讀excel,xlwt是寫excel的庫。 今天就先來說一下xl ......

    uj5u.com 2020-09-10 05:36:28 more
  • 當我們創建HashMap時,底層到底做了什么?

    jdk1.7中的底層實作程序(底層基于陣列+鏈表) 在我們new HashMap()時,底層創建了默認長度為16的一維陣列Entry[ ] table。當我們呼叫map.put(key1,value1)方法向HashMap里添加資料的時候: 首先,呼叫key1所在類的hashCode()計算key1 ......

    uj5u.com 2020-09-10 05:36:38 more
最新发布
  • 【中介者設計模式詳解】C/Java/JS/Go/Python/TS不同語言實作

    * 中介者模式是一種行為型設計模式,它可以用來減少類之間的直接依賴關系,
    * 將物件之間的通信封裝到一個中介者物件中,從而使得各個物件之間的關系更加松散。
    * 在中介者模式中,物件之間不再直接相互互動,而是通過中介者來中轉訊息。 ......

    uj5u.com 2023-04-20 08:20:47 more
  • 露天煤礦現場調研和交流案例分享

    他們集團的資訊化公司及研究院在一個礦區正在做智能礦山的統一平臺的 試點,專案投資大概1億,包括了礦山的各方面的內容,顯示得我們這次交流有點多余。他們2年前開始做智能礦山的規劃,有很多煤礦行業專家的加持,他們的描述是非常完美,但是去年底應該上線的平臺,現在還沒有看到影子。他們確實有很多場景需求,但是被... ......

    uj5u.com 2023-04-20 08:20:25 more
  • 《社區人員管理》實戰案例設計&個人案例分享

    設計是一個讓人夢想成真程序,開始編碼、測驗、除錯之前進行需求分析和架構設計,才能保證關鍵方面都做正確 ......

    uj5u.com 2023-04-20 08:20:17 more
  • 軟體架構生態化-多角色交付的探索實踐

    作為一個技術架構師,不僅僅要緊跟行業技術趨勢,還要結合研發團隊現狀及痛點,探索新的交付方案。在日常中,你是否遇到如下問題 “ 業務需求排期長研發是瓶頸;非研發角色感受不到研發技改提效的變化;引入ISV 團隊又擔心質量和安全,培訓周期長“等等,基于此我們探索了一種新的技術體系及交付方案來解決如上問題。 ......

    uj5u.com 2023-04-20 08:20:10 more
  • 【中介者設計模式詳解】C/Java/JS/Go/Python/TS不同語言實作

    * 中介者模式是一種行為型設計模式,它可以用來減少類之間的直接依賴關系,
    * 將物件之間的通信封裝到一個中介者物件中,從而使得各個物件之間的關系更加松散。
    * 在中介者模式中,物件之間不再直接相互互動,而是通過中介者來中轉訊息。 ......

    uj5u.com 2023-04-20 08:19:44 more
  • 露天煤礦現場調研和交流案例分享

    他們集團的資訊化公司及研究院在一個礦區正在做智能礦山的統一平臺的 試點,專案投資大概1億,包括了礦山的各方面的內容,顯示得我們這次交流有點多余。他們2年前開始做智能礦山的規劃,有很多煤礦行業專家的加持,他們的描述是非常完美,但是去年底應該上線的平臺,現在還沒有看到影子。他們確實有很多場景需求,但是被... ......

    uj5u.com 2023-04-20 08:19:07 more
  • 《社區人員管理》實戰案例設計&個人案例分享

    設計是一個讓人夢想成真程序,開始編碼、測驗、除錯之前進行需求分析和架構設計,才能保證關鍵方面都做正確 ......

    uj5u.com 2023-04-20 08:18:57 more
  • 軟體架構生態化-多角色交付的探索實踐

    作為一個技術架構師,不僅僅要緊跟行業技術趨勢,還要結合研發團隊現狀及痛點,探索新的交付方案。在日常中,你是否遇到如下問題 “ 業務需求排期長研發是瓶頸;非研發角色感受不到研發技改提效的變化;引入ISV 團隊又擔心質量和安全,培訓周期長“等等,基于此我們探索了一種新的技術體系及交付方案來解決如上問題。 ......

    uj5u.com 2023-04-20 08:18:49 more
  • 05單件模式

    #經典的單件模式 public class Singleton { private static Singleton uniqueInstance; //一個靜態變數持有Singleton類的唯一實體。 // 其他有用的實體變數寫在這里 //構造器宣告為私有,只有Singleton可以實體化這個類! ......

    uj5u.com 2023-04-19 08:42:51 more
  • 【架構與設計】常見微服務分層架構的區別和落地實踐

    軟體工程的方方面面都遵循一個最基本的道理:沒有銀彈,架構分層模型更是如此,每一種都有各自優缺點,所以請根據不同的業務場景,并遵循簡單、可演進這兩個重要的架構原則選擇合適的架構分層模型即可。 ......

    uj5u.com 2023-04-19 08:42:41 more