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下实现农历日历的代码
Mar 07 PHP
php开发工具之vs2005图解
Jan 12 PHP
php在字符串中查找另一个字符串
Nov 19 PHP
PHP可变函数的使用详解
Jun 14 PHP
PHP实现数字补零功能的2个函数介绍
May 12 PHP
php实现判断访问来路是否为搜索引擎机器人的方法
Apr 15 PHP
thinkphp,onethink和thinkox中验证码不显示的解决方法分析
Jun 06 PHP
php实现网页端验证码功能
Jul 11 PHP
php双层循环(九九乘法表)
Oct 23 PHP
PHP使用Redis长连接的方法详解
Feb 12 PHP
PHP中soap用法示例【SoapServer服务端与SoapClient客户端编写】
Dec 25 PHP
PHP实现的62进制转10进制,10进制转62进制函数示例
Jun 06 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的MVC模式实现原理分析(一相简单的MVC框架范例)
2014/04/29 PHP
Yii2验证器(Validator)用法分析
2016/07/23 PHP
PHP操作XML中XPath的应用示例
2019/07/04 PHP
laravel ORM关联关系中的 with和whereHas用法
2019/10/16 PHP
Laravel 实现在Blade模版中使用全局变量代替路径的例子
2019/10/22 PHP
Extjs学习笔记之五 一个小细节renderTo和applyTo的区别
2010/01/07 Javascript
今天是星期几的4种JS代码写法
2013/09/17 Javascript
js阻止事件追加的具体实现
2014/10/15 Javascript
js实现iframe跨页面调用函数的方法
2014/12/13 Javascript
JSONP之我见
2015/03/24 Javascript
JS右下角广告窗口代码(可收缩、展开及关闭)
2015/09/04 Javascript
快速学习jQuery插件 jquery.validate.js表单验证插件使用方法
2015/12/01 Javascript
详解Sea.js中Module.exports和exports的区别
2017/02/12 Javascript
vue如何使用 Slot 分发内容实例详解
2017/09/05 Javascript
vue项目打包部署_nginx代理访问方法详解
2018/09/20 Javascript
JavaScript表格隔行变色和Tab标签页特效示例【附jQuery版】
2019/07/11 jQuery
Vue插件之滑动验证码
2019/09/21 Javascript
[02:14]DOTA2英雄基础教程 修补匠
2013/12/23 DOTA
[03:55]显微镜下的DOTA2特别篇——430灰烬之灵神级操作
2014/06/24 DOTA
[01:00:14]DOTA2官方TI8总决赛纪录片 真视界True Sight
2019/01/16 DOTA
python基础教程之类class定义使用方法
2014/02/20 Python
python中的代码编码格式转换问题
2015/06/10 Python
Python常用时间操作总结【取得当前时间、时间函数、应用等】
2017/05/11 Python
详解Python安装scrapy的正确姿势
2018/06/26 Python
Scrapy框架使用的基本知识
2018/10/21 Python
python变量赋值方法(可变与不可变)
2019/01/12 Python
Python 一键获取百度网盘提取码的方法
2019/08/01 Python
将tf.batch_matmul替换成tf.matmul的实现
2020/06/18 Python
使用canvas压缩图片大小的方法示例
2019/08/02 HTML / CSS
日本面向世界,国际级的免税在线购物商城:DOKODEMO
2017/02/01 全球购物
大学同学聚会邀请函
2014/01/29 职场文书
新年主持词
2014/03/27 职场文书
2015年大学生入党自荐书
2015/03/24 职场文书
公司搬迁通知
2015/04/20 职场文书
防汛通知
2015/04/25 职场文书
2016应届毕业生实习心得体会
2015/10/09 职场文书