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学习之数组值的操作
Apr 17 PHP
ThinkPHP采用模块和操作分析
Apr 18 PHP
PHP 利用AJAX获取网页并输出的实现代码(Zjmainstay)
Aug 31 PHP
php使用strtotime和date函数判断日期是否有效代码分享
Dec 25 PHP
PHP判断远程图片或文件是否存在的实现代码
Feb 20 PHP
php实现专业获取网站SEO信息类实例
Apr 02 PHP
php基于Snoopy解析网页html的方法
Jul 09 PHP
php利用smtp类实现电子邮件发送
Oct 30 PHP
php+mysql实现的二级联动菜单效果详解
May 10 PHP
PHP GD库相关图像生成和处理函数小结
Sep 30 PHP
cakephp2.X多表联合查询join及使用分页查询的方法
Feb 23 PHP
PHP用函数嵌入网站访问量计数器
Oct 27 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
基于xcache的配置与使用详解
2013/06/18 PHP
又拍云异步上传实例教程详解
2016/04/19 PHP
php is_executable判断给定文件名是否可执行实例
2016/09/26 PHP
原生PHP实现导出csv格式Excel文件的方法示例【附源码下载】
2019/03/07 PHP
javascript 支持链式调用的异步调用框架Async.Operation
2009/08/04 Javascript
原生js拖拽(第一课 未兼容)拖拽思路
2013/03/29 Javascript
JS+css 图片自动缩放自适应大小
2013/08/08 Javascript
jQuery中fadeOut()方法用法实例
2014/12/24 Javascript
javaScript实现滚动新闻的方法
2015/07/30 Javascript
详解Bootstrap四种图片样式
2016/01/04 Javascript
微信小程序 限制1M的瘦身技巧与方法详解
2017/01/06 Javascript
jQuery实现简单漂亮的Nav导航菜单效果
2017/03/29 jQuery
angularjs $http实现form表单提交示例
2017/06/09 Javascript
vue-cli中的babel配置文件.babelrc实例详解
2018/02/22 Javascript
Vue中的vue-resource示例详解
2018/11/02 Javascript
Vue+Element UI+vue-quill-editor富文本编辑器及插入图片自定义
2019/08/20 Javascript
vue日历/日程提醒/html5本地缓存功能
2019/09/02 Javascript
layui的select联动实现代码
2019/09/28 Javascript
vue.js循环radio的实例
2019/11/07 Javascript
解决在Vue中使用axios POST请求变成OPTIONS的问题
2020/08/14 Javascript
vue 项目引入echarts 添加点击事件操作
2020/09/09 Javascript
[42:06]2019国际邀请赛全明星赛 8.23
2019/09/05 DOTA
python实现从网络下载文件并获得文件大小及类型的方法
2015/04/28 Python
Django中数据库的数据关系:一对一,一对多,多对多
2018/10/21 Python
Python动态赋值的陷阱知识点总结
2019/03/17 Python
python2与python3爬虫中get与post对比解析
2019/09/18 Python
python实现提取str字符串/json中多级目录下的某个值
2020/02/27 Python
解决Python spyder显示不全df列和行的问题
2020/04/20 Python
在Sublime Editor中配置Python环境的详细教程
2020/05/03 Python
CSS3中的content属性使用示例
2015/07/20 HTML / CSS
请写出 BOOL flag 与"零值"比较的 if 语句
2016/02/29 面试题
2014婚礼司仪主持词
2014/03/14 职场文书
镇政府副镇长群众路线专题民主生活会对照检查材料
2014/09/19 职场文书
学生会工作感言
2015/08/07 职场文书
优化经济发展环境工作总结
2015/08/11 职场文书
幼儿园体操比赛口号
2015/12/25 职场文书