首頁  >  文章  >  php教程  >  PHP物件導向之旅:深入理解static變數與方法

PHP物件導向之旅:深入理解static變數與方法

黄舟
黄舟原創
2016-12-12 17:06:071384瀏覽

static關鍵字聲明一個屬性或方法是和類別相關的,而不是和類別的某個特定的實例相關,因此,這類屬性或方法也稱為「類別屬性」或「類別方法」。

如果存取控制權限允許,可不必建立該類別物件而直接使用類別名稱加上兩個冒號「::」呼叫。

static關鍵字可以用來修飾變數、方法。

不經過實例化,就可以直接存取類別中static的屬性和static的方法。

static 的屬性和方法,只能存取static的屬性和方法,不能類別存取非靜態的屬性和方法。因為靜態屬性和方法被創建時,可能還沒有任何這個類別的實例可以被呼叫。

static的屬性,在記憶體中只有一份,為所有的實例共用。

使用self:: 關鍵字存取目前類別的靜態成員。
靜態屬性公用特性

一個類別的所有實例,共用類別中的靜態屬性。

也就是說,在記憶體中即使有多個實例,靜態的屬性也只有一份。

下面範例中的設定了一個計數器$count屬性,設定private 和 static 修飾。這樣,外界並不能直接存取$count屬性。而程式運行的結果我們也看到多個實例在使用同一個靜態的$count 屬性。

<?
class user{
    private static $count = 0 ; //记录所有用户的登录情况.
    public function __construct(){
        self::$count = self::$count + 1;
    }
    public function getCount(){    
        return self::$count;
    }
    public function __destruct(){
        self::$count = self::$count -1;
    }
}
$user1 = new user();
$user2 = new user();
$user3 = new user();
echo "now here have ".$user1->getCount()." user";
echo "<br>";
unset( $user3);
echo "now here have ".$user1->getCount()." user";
?>

程式運作結果:
1
2

now here have 3 user
now here have 2 user jb51.net
靜態屬性直接呼叫

靜態屬性不需要實例化就可以直接使用,在類別還沒創建時就可以直接使用。

使用的方式是 類別名稱::靜態屬性名稱。

<?
class Math{
    public static $pi = 3.14;

}
//求一个半径3的园的面积。
$r = 3;
echo "半径是 $r 的面积是<br>";
echo Math::$pi * $r * $r ;

echo "<br><br>";
//这里我觉得 3.14 不够精确,我把它设置的更精确。
Math::$pi = 3.141592653589793;
echo "半径是 $r 的面积是<br>";
echo Math::$pi * $r * $r ;
?>

程式運作結果:
1
2
3
4

半徑是 3 的面積是
28.26
半徑是 3 的面積是
28.2743338823

類別沒有創建,靜態屬性就可以直接使用。那靜態屬性在什麼時候在記憶體中被創建?在PHP中沒有看到相關的資料。引用Java中的概念,來解釋應該也具有通用性。

靜態屬性和方法,在類別被呼叫時創建。類別被調用,是指類別被創建或類別中的任何靜態成員被調用。
靜態方法

靜態方法不需要所在類別被實例化就可以直接使用。

使用的方式是 類別名稱::靜態方法名稱。

下面我們繼續寫這個Math類,用來進行數學計算。我們設計一個方法用來算出其中的最大值。既然是數學運算,我們也沒有必要去實例化這個類,如果這個方法可以拿過來就用就方便多了。

我們這只是為了示範static方法而設計的這個類別。在PHP提供了 max() 函數比較數值。

<?
class Math{

    public static function Max($num1,$num2){
        return $num1 > $num2 ? $num1 : $num2;
    }    
}
$a = 99;
$b = 88;
echo "显示 $ a 和 $ b 中的最大值是";
echo "<br>";
echo Math::Max($a,$b);
echo "<br>";echo "<br>";echo "<br>";
$a = 99;
$b = 100;
echo "显示 $ a 和 $ b 中的最大值是";
echo "<br>";
echo Math::Max($a,$b);
?>

程式運作結果:

顯示 $ a 和 $ b 中的最大值是
99
顯示 $ a 和 $ b 中的最大值是
100
靜態方法如何呼叫靜態方法

第一個例子,一個靜態方法呼叫其它靜態方法時,直接使用 類別名稱。

<?
// 实现最大值比较的Math类。
class Math{

    public static function Max($num1,$num2){
        return $num1 > $num2 ? $num1 : $num2;
    }
    public static function Max3($num1,$num2,$num3){
       $num1 = Math::Max($num1,$num2);
       $num2 = Math::Max($num2,$num3);
       $num1 = Math::Max($num1,$num2);       
       return $num1;
    }
}
$a = 99;
$b = 77;
$c = 88;
echo "显示 $a  $b $c  中的最大值是";
echo "<br>";
echo Math::Max3($a,$b,$c);
?>

程式運行結果:
1
2

顯示 99 77 88 中的最大值是
99

也可以使用self:: 調用當前類別中的其靜態方法。 (建議)

<?
// 实现最大值比较的Math类。
class Math{

    public static function Max($num1,$num2){
        return $num1 > $num2 ? $num1 : $num2;
    }
    public static function Max3($num1,$num2,$num3){
       $num1 = self::Max($num1,$num2);
       $num2 = self::Max($num2,$num3);
       $num1 = self::Max($num1,$num2);       
       return $num1;
    }
}
$a = 99;
$b = 77;
$c = 88;
echo "显示 $a  $b $c  中的最大值是";
echo "<br>";
echo Math::Max3($a,$b,$c);
?>

程式運作結果:
1
2

顯示99 77 88 中的最大值是
99
靜態方法呼叫靜態屬性

使用類別的靜態屬性名稱。

<?
//
class Circle{
    public static $pi = 3.14;

    public static function circleAcreage($r){
      return $r * $r * Circle::$pi;
    }
}
$r = 3;
echo " 半径 $r 的圆的面积是 " . Circle::circleAcreage($r);
?>

程式運作結果:
1

半徑 3 的圓的面積是 28.26

使用self:: 呼叫本類的靜態屬性。 (建議)

<?
//
class Circle{
    public static $pi = 3.14;

    public static function circleAcreage($r){
      return $r * $r * self::$pi;
    }
}
$r = 3;
echo " 半径 $r 的圆的面积是 " . Circle::circleAcreage($r);
?>

程式運作結果:
1

半徑 3 的圓的面積是 28.26
靜態方法不能呼叫非靜態屬性

靜態方法不能呼叫非靜態的屬性。不能使用self::呼叫非靜態屬性。

<?
//
class Circle{
    public $pi = 3.14;

    public static function circleAcreage($r){
      return $r * $r * self::pi;
    }
}
$r = 3;
echo " 半径 $r 的圆的面积是 " . Circle::circleAcreage($r);
?>

程式運行結果:
1

Fatal error: Undefined class constant 'pi' in E:PHPProjectstest.php on line 7

也不能使用 $this 取得非靜態屬性的值。

<?
//
class Circle{
    public $pi = 3.14;

    public static function circleAcreage($r){
      return $r * $r * $this->pi;
    }
}
$r = 3;
echo " 半径 $r 的圆的面积是 " . Circle::circleAcreage($r);
?>

程式運作結果:
1

Fatal error: Using $this when not in object context in E:PHPProjectstest.php on line 7
靜態方法呼叫非靜態方法

PHP5中,在靜態方法中不能使用 $this 標識呼叫非靜態方法。

<?
// 实现最大值比较的Math类。
class Math{   
    public function Max($num1,$num2){
        echo "bad<br>";       
        return $num1 > $num2 ? $num1 : $num2;
    }
    public static function Max3($num1,$num2,$num3){
       $num1 = $this->Max($num1,$num2);
       $num2 = $this->Max($num2,$num3);
       $num1 = $this->Max($num1,$num2);       
       return $num1;
    }
}
$a = 99;
$b = 77;
$c = 188;
echo "显示 $a  $b $c  中的最大值是";
echo "<br>";
echo Math::Max3($a,$b,$c);
?>

程式運作結果:

顯示 99 77 188 中的最大值是
Fatal error: Using $this when not in object context in E:test.php on line 10

當一個類別中有非靜態方法被 self:: 呼叫時,系統會自動將這個方法轉換為靜態方法。

下面的程式碼被執行了,而且有結果。因為Max方法被系統轉換成靜態方法了。

<?
// 实现最大值比较的Math类。
class Math{   
    public function Max($num1,$num2){      
        return $num1 > $num2 ? $num1 : $num2;
    }
    public static function Max3($num1,$num2,$num3){
       $num1 = self::Max($num1,$num2);
       $num2 = self::Max($num2,$num3);
       $num1 = self::Max($num1,$num2);       
       return $num1;
    }
}
$a = 99;
$b = 77;
$c = 188;
echo "显示 $a  $b $c  中的最大值是";
echo "<br>";
echo Math::Max3($a,$b,$c);
?>

程式運作結果:
1
2

顯示99 77 188 中的最大值是
188

下面的例子中,我們讓靜態方法Max3 用過self::::了非靜態方法Max,有讓下面的範例中,我們讓靜態方法Max3 用過self::::了非靜態方法Max,有讓非靜態方法Max透過$this 呼叫非靜態屬性$pi 。

在運作是報出了錯誤,這個錯誤和前一個範例 3-1-9.php一樣,這次倒是非靜態方法Max報出了靜態方法呼叫非靜態屬性的錯誤。

這倒是證明了一點,在這裡我們定義的非靜態方法 Max 被系統自動轉換成靜態方法了。

<?
// 实现最大值比较的Math类。
class Math{
    public $pi = 3.14;

    public function Max($num1,$num2){
        echo self::$pi;  //这里的调用看来不应该有问题.
        return $num1 > $num2 ? $num1 : $num2;
    }
    public static function Max3($num1,$num2,$num3){
       $num1 = self::Max($num1,$num2);
       $num2 = self::Max($num2,$num3);
       $num1 = self::Max($num1,$num2);       
       return $num1;
    }
}
$a = 99;
$b = 77;
$c = 188;
echo "显示 $a  $b $c  中的最大值是";
echo "<br>";
echo Math::Max3($a,$b,$c);
?>

程序运行结果:
1
2

显示 99 77 188 中的最大值是
Fatal error: Access to undeclared static property: Math::$pi in E:PHPProjectstest.php on line 7


陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn