- 相關(guān)推薦
PHP對象相關(guān)知識總結(jié)
文章主要介紹了PHP對象相關(guān)知識總結(jié)的相關(guān)資料,需要的朋友可以參考下,就跟隨百分網(wǎng)小編一起去了解下吧,想了解更多相關(guān)信息請持續(xù)關(guān)注我們應(yīng)屆畢業(yè)生考試網(wǎng)!
對象傳遞:一種說法是“PHP對象是通過引用傳遞的”,更準(zhǔn)確的說法是別名(標(biāo)識符)傳遞,即它們都保存著同一個標(biāo)識符(ID)的拷貝,這個標(biāo)識符指向同一個對象的真正內(nèi)容。
<?php
class A {
public $foo = 1;
}
$a = new A;
$b = $a; // $a ,$b都是同一個標(biāo)識符的拷貝
// ($a) = ($b) = <id>
$b->foo = 2;
echo $a->foo."\n";//2
$c = new A;
$d = &$c; // $c ,$d是引用
// ($c,$d) = <id>
$d->foo = 2;
echo $c->foo."\n";//2
$e = new A;
function foo($obj) {
// ($obj) = ($e) = <id>
$obj->foo = 2;
}
foo($e);
echo $e->foo."\n";//2
?對象復(fù)制:對象復(fù)制可以通過 clone 關(guān)鍵字來完成,如果原對象定義了 __clone() 方法,則新對象中的 __clone() 方法將在復(fù)制完后被調(diào)用,__clone() 方法可用于修改復(fù)制對象屬性的值。當(dāng)對象被復(fù)制后,會對對象的所有屬性執(zhí)行一個淺復(fù)制(shallow copy),但所有的引用屬性仍然會是一個指向原來的變量的引用。
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
<?php
class SubObject
{
static $instances = 0;
public $instance;
public function __construct()
{
$this->instance = ++self::$instances;
}
public function __clone()
{
$this->instance = ++self::$instances;
}
}
class MyCloneable
{
public $object1;
public $object2;
function __clone()
{
// 強(qiáng)制復(fù)制一份this->object, 否則仍然指向同一個對象
$this->object1 = clone $this->object1;
}
function cloneTest()
{
echo 'cloneTest';
}
}
$obj = new MyCloneable();
$obj->object1 = new SubObject();
$obj->object2 = new SubObject();
$obj2 = clone $obj;
print("Original Object:\n");
print_r($obj);
print("Cloned Object:\n");
print_r($obj2);
echo $obj2->cloneTest().":\n";
echo (new Reflectionclass($obj2));
上例輸出結(jié)果:
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
Original Object:
MyCloneable Object
(
[object1] => SubObject Object
(
[instance] => 1
)
[object2] => SubObject Object
(
[instance] => 2
)
)
Cloned Object:
MyCloneable Object
(
[object1] => SubObject Object
(
[instance] => 3
)
[object2] => SubObject Object
(
[instance] => 2
)
)
cloneTest:
Class [ <user> class MyCloneable ] {
@@ /public/t.php 18-33
- Constants [0] {
}
- Static properties [0] {
}
- Static methods [0] {
}
- Properties [2] {
Property [ <default> public $object1 ]
Property [ <default> public $object2 ]
}
- Methods [2] {
Method [ <user> public method __clone ] {
@@ /public/t.php 23 - 27
}
Method [ <user> public method cloneTest ] {
@@ /public/t.php 29 - 32
}
}
}
?對象遍歷: foreach只能遍歷對象的可見屬性,無法遍歷其方法,實(shí)現(xiàn)起來比較容易;另外,也可通過實(shí)現(xiàn)Iterator接口或IteratorAggregate接口的方法遍歷對象屬性。
?類型約束: PHP作為一種弱類型語言,類型約束可以讓編程更加規(guī)范,也少出些差錯;類型約束不只能用在對象定義中,也能用在函數(shù)定義中。類型約束可指定對象、接口、array、callable(閉包c(diǎn)allback),類型約束用來保證實(shí)際數(shù)據(jù)類型與原型定義一致,不一致則拋出一個可捕獲的致命錯誤;不過如果定義了默認(rèn)值為NULL,那么實(shí)參可以是NULL;類型約束不能用于標(biāo)量類型如 int 或 string,Traits 也不允許。
?對象序列化與還原:函數(shù)serialize()可將打成包含字節(jié)流的字符串便于存儲對象,函數(shù)unserialize()能夠還原字符串為對象。但有一個前提是,無論序列化還是反序列化,對象的類定義已經(jīng)完成,即需要先導(dǎo)入類(文件)。
?重載:PHP的重載包括屬性和方法,更像一個套用說法,不支持常見的重載語法規(guī)范,具有不可預(yù)見性,影響范圍更寬泛,就是利用魔術(shù)方法(magic methods)來調(diào)用當(dāng)前環(huán)境下未定義或不可見的類屬性或方法。所有重載方法都必須被聲明為 public(這一條應(yīng)該比較好理解,別人可能因不可見才需要你,那你自己必須可見才行),參數(shù)也不能通過引用傳遞(重載方法具有不可預(yù)見性,估計出于安全方面的考慮吧,防止變量被隨意引用)。在除 isset() 外的其它語言結(jié)構(gòu)中無法使用重載的屬性,這意味著當(dāng)對一個重載的屬性使用 empty() 時,重載魔術(shù)方法將不會被調(diào)用; 為避開此限制,必須將重載屬性賦值到本地變量再使用 empty(),可見重載屬性是介于合法屬性與非法屬性之間的存在。
[屬性重載]:這些方法不能被聲明為 static,在靜態(tài)方法中,這些魔術(shù)方法將不會被調(diào)用
public void __set ( string $name , mixed $value )
在給不可訪問屬性賦值時,__set() 會被調(diào)用
public mixed __get ( string $name )
讀取不可訪問屬性的值時,__get() 會被調(diào)用
public bool __isset ( string $name )
當(dāng)對不可訪問屬性調(diào)用 isset() 或 empty() 時,__isset() 會被調(diào)用
public void __unset ( string $name )
當(dāng)對不可訪問屬性調(diào)用 unset() 時,__unset() 會被調(diào)用
Note:
因為 PHP 處理賦值運(yùn)算的方式,__set() 的返回值將被忽略。類似的, 在下面這樣的鏈?zhǔn)劫x值中,__get() 不會被調(diào)用:
$a = $obj->b = 8;
[方法重載]:
public mixed __call ( string $name , array $arguments )
在對象中調(diào)用一個不可訪問方法時,__call() 會被調(diào)用
public static mixed __callStatic ( string $name , array $arguments )
在靜態(tài)上下文中調(diào)用一個不可訪問方法時,__callStatic() 會被調(diào)用
?靜態(tài)屬性和方法:static 關(guān)鍵字用來定義靜態(tài)屬性、靜態(tài)方法,靜態(tài)屬性不能通過實(shí)例化的對象-> 來訪問(但靜態(tài)方法可以)。靜態(tài)屬性只能被初始化為常量表達(dá)式,所以可以把靜態(tài)屬性初始化為整數(shù)或數(shù)組,但不能初始化為另一個變量或函數(shù)返回值,也不能指向一個對象。可以用一個變量表示類來動態(tài)調(diào)用靜態(tài)屬性,但該變量的值不能為關(guān)鍵字 self,parent 或 static。
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
class Foo
{
public static $my_static = 'foo';
public function staticValue() {
return self::$my_static;
}
}
class Bar extends Foo
{
public function fooStatic() {
return parent::$my_static;
}
}
print Foo::$my_static . "\n";
$foo = new Foo();
print $foo->staticValue() . "\n";
print $foo->my_static . "\n"; // Undefined "Property" my_static
print $foo::$my_static . "\n";
$classname = 'Foo';
print $classname::$my_static . "\n"; // As of PHP 5.3.0
print Bar::$my_static . "\n";
$bar = new Bar();
print $bar->fooStatic() . "\n";
?后期靜態(tài)綁定:static:: 定義后期靜態(tài)綁定工作原理是存儲了上一個“非轉(zhuǎn)發(fā)調(diào)用”(non-forwarding call)的類名。當(dāng)進(jìn)行靜態(tài)方法調(diào)用時,該類名即為明確指定的那個(通常在 :: 運(yùn)算符左側(cè)部分);當(dāng)進(jìn)行非靜態(tài)方法調(diào)用時,即為該對象所屬的類。使用 self:: 或者 __CLASS__ 對當(dāng)前類的靜態(tài)引用,取決于定義當(dāng)前方法所在的類;static:: 不再被解析為定義當(dāng)前方法所在的類,而是在實(shí)際運(yùn)行時計算的,可以用于靜態(tài)屬性和所有方法的調(diào)用。
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
<?php
class A
{
private $proPrivate = "private of A";
protected $proProtected = "protected of A";
public $proPublic = "public of A";
private function foo()
{
echo $this->proPrivate."\n";
echo $this->proProtected."\n";
echo $this->proPublic."\n";
}
public function test()
{
$this->foo();
static::foo();
}
}
class B extends A
{
/* foo() will be copied to B, hence its scope will still be A and
* the call be successful */
}
class C extends A
{
private $proPrivate = "private of C";
protected $proProtected = "protected of C";
public $proPublic = "public of C";
private function foo()
{
/* original method is replaced; the scope of the new one is C */
echo "I am C\n";
}
public function myFoo()
{
//parent::foo();
$this->foo();
}
}
echo "Class B:\n";
$b = new B();
$b->test();
echo "\nClass C:\n";
$c = new C();
$c->myFoo();
$c->test(); //fails
上例輸出結(jié)果:
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
Class B:
private of A
protected of A
public of A
private of A
protected of A
public of A
Class C:
I am C
private of A
protected of C
public of C
Fatal error: Uncaught Error: Call to private method C::foo() from context 'A' in /public/t.php:19 Stack trace: #0 /public/t.php(54): A->test() #1 {main} thrown in /public/t.php on line 19
?繼承:官方文檔對繼承有這樣一段描述“當(dāng)擴(kuò)展一個類,子類就會繼承父類所有公有的和受保護(hù)的方法。除非子類覆蓋了父類的方法,被繼承的方法都會保留其原有功能”,言下之意似乎私有屬性和方法不會被繼承;然而上例又告訴我們子類擁有與父類一致的屬性和方法,繼承就是全盤復(fù)制,這才能滿足我們對繼承編程的需求,如果私有的不能繼承,子類就必須自行重新定義,在大多數(shù)時候沒有必要。另外就是可見性問題,父類的私有屬性和方法在子類是不可見的。上例還告訴我們對象實(shí)際執(zhí)行的域要考慮可見性、繼承、后期靜態(tài)綁定機(jī)制。
【PHP對象相關(guān)知識總結(jié)】相關(guān)文章:
PHP類和對象的相關(guān)函數(shù)講解07-10
PHP中的類與對象入門知識09-05
java類與對象的相關(guān)知識10-02
PHP函數(shù)知識總結(jié)09-29
PHP對象Object的概念詳解09-05
PHP語言面向?qū)ο蟮膬?nèi)容06-22