PHP面向对象之旅:深入理解static变量与方法


Posted in PHP onJanuary 06, 2014

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 3water.com
静态属性直接调用

静态属性不需要实例化就可以直接使用,在类还没有创建时就可以直接使用。

使用的方式是 类名::静态属性名。

<?
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,有让非静态方法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

PHP 相关文章推荐
php中的时间显示
Jan 18 PHP
一周让你学会PHP 不错的学习资料
Feb 06 PHP
php学习笔记(三)操作符与控制结构
Aug 06 PHP
php Session存储到Redis的方法
Nov 04 PHP
PHP文件上传判断file是否己选择上传文件的方法
Nov 10 PHP
linux中cd命令使用详解
Jan 08 PHP
typecho插件编写教程(二):写一个新插件
May 28 PHP
php轻量级的性能分析工具xhprof的安装使用
Aug 12 PHP
php socket通信(tcp/udp)实例分析
Feb 14 PHP
PHP入门教程之会话控制技巧(cookie与session)
Sep 11 PHP
PHP简单数据库操作类实例【支持增删改查及链式操作】
Oct 10 PHP
PHP实现绘制二叉树图形显示功能详解【包括二叉搜索树、平衡树及红黑树】
Nov 16 PHP
浅析ThinkPHP中的pathinfo模式和URL重写
Jan 06 #PHP
php Calender(日历)代码分享
Jan 03 #PHP
深入解读php中关于抽象(abstract)类和抽象方法的问题分析
Jan 03 #PHP
PHP运行SVN命令显示某用户的文件更新记录的代码
Jan 03 #PHP
PHP抓屏函数实现屏幕快照代码分享
Jan 02 #PHP
php curl模拟post提交数据示例
Dec 31 #PHP
codeigniter使用技巧批量插入数据实例方法分享
Dec 31 #PHP
You might like
php生成SessionID和图片校验码的思路和实现代码
2009/03/10 PHP
php 进度条实现代码
2009/03/10 PHP
Windows Apache2.2.11及Php5.2.9-1的安装与配置方法
2009/06/08 PHP
php保存信息到当前Session的方法
2015/03/16 PHP
详解WordPress中调用评论模板和循环输出评论的PHP函数
2016/01/05 PHP
php无限级分类实现方法分析
2016/10/19 PHP
prototype1.4中文手册
2006/09/22 Javascript
通过ifame指向的页面高度调整iframe的高度
2006/10/05 Javascript
Js与下拉列表处理问题解决
2014/02/13 Javascript
深入理解JS中的substr和substring
2016/04/26 Javascript
web 前端常用组件之Layer弹出层组件
2016/09/22 Javascript
详解用node.js实现简单的反向代理
2017/06/26 Javascript
JS实现图片手风琴效果
2020/04/17 Javascript
微信小程序媒体组件详解(视频,音乐,图片)
2017/09/19 Javascript
vue使用v-if v-show页面闪烁,div闪现的解决方法
2018/10/12 Javascript
angular6 填坑之sdk的方法
2018/12/27 Javascript
JavaScript Window浏览器对象模型原理解析
2020/05/30 Javascript
Vue中nprogress页面加载进度条的方法实现
2020/11/13 Javascript
简单讲解Python中的数字类型及基本的数学计算
2016/03/11 Python
python利用OpenCV2实现人脸检测
2020/04/16 Python
python构建深度神经网络(续)
2018/03/10 Python
django 使用 request 获取浏览器发送的参数示例代码
2018/06/11 Python
Django 实现Admin自动填充当前用户的示例代码
2019/11/18 Python
python使用PIL剪切和拼接图片
2020/03/23 Python
Python操作Word批量生成合同的实现示例
2020/08/28 Python
Python中全局变量和局部变量的理解与区别
2021/02/07 Python
使用CSS3来匹配横屏竖屏的简单方法
2015/08/04 HTML / CSS
意大利大型购物中心:Oliviero.it
2017/10/19 全球购物
购买美国制造的相框和画框架:Picture Frames
2018/08/14 全球购物
瑞典多品牌连锁店:Johnells
2021/01/13 全球购物
岗位说明书范文
2014/05/07 职场文书
解放思想大讨论活动总结
2015/05/09 职场文书
公司要求试用期员工提交“述职报告”,该怎么写?
2019/07/17 职场文书
Golang 空map和未初始化map的注意事项说明
2021/04/29 Golang
详解Vue slot插槽
2021/11/20 Vue.js
Debian11 Xfce终端光标的颜色怎么设置?
2022/08/14 数码科技