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 相关文章推荐
excellent!――ASCII Art(由目标图象生成ascii)
Feb 20 PHP
Ajax+PHP边学边练 之五 图片处理
Dec 03 PHP
php生成随机密码的三种方法小结
Sep 04 PHP
使用cookie实现统计访问者登陆次数
Jun 08 PHP
关于url地址传参数时字符串有回车造成页面脚本赋值失败的解决方法
Jun 28 PHP
百度实时推送api接口应用示例
Oct 21 PHP
PHP使用feof()函数读文件的方法
Nov 07 PHP
PHP实现批量修改文件后缀名的方法
Jul 30 PHP
php对二维数组进行相关操作(排序、转换、去空白等)
Nov 04 PHP
PHP代码维护,重构变困难的4种原因分析
Jan 25 PHP
用PHP的反射实现委托模式的讲解
Mar 22 PHP
PHP文件操作简单介绍及函数汇总
Dec 11 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图片处理之图片旋转和图片翻转实例
2014/11/19 PHP
php使用Jpgraph绘制复杂X-Y坐标图的方法
2015/06/10 PHP
php 多文件上传的实现实例
2016/10/23 PHP
PHP实现简单用户登录界面
2019/10/23 PHP
对采用动态原型方式无法展示继承机制得思考
2009/12/04 Javascript
Jquery判断IE6等浏览器的代码
2011/04/05 Javascript
javascript查找字符串中出现最多的字符和次数的小例子
2013/10/29 Javascript
初识angular框架后的所思所想
2016/02/19 Javascript
阿里云ecs服务器中安装部署node.js的步骤
2016/10/08 Javascript
Vue2 SSR渲染根据不同页面修改 meta
2017/11/20 Javascript
ReactNative 之FlatList使用及踩坑封装总结
2017/11/29 Javascript
详解mpvue小程序中怎么引入iconfont字体图标
2018/10/01 Javascript
[03:24]CDEC.Y赛前采访 努力备战2016国际邀请赛中国区预选赛
2016/06/25 DOTA
[00:57]深扒TI7聊天轮盘语音出处5
2017/05/11 DOTA
[03:12]TI9战队档案 - Virtus Pro
2019/08/20 DOTA
[01:20:47]DOTA2-DPC中国联赛 正赛 Ehome vs Magma BO3 第一场 1月19日
2021/03/11 DOTA
讲解Python中的标识运算符
2015/05/14 Python
pandas数据处理基础之筛选指定行或者指定列的数据
2018/05/03 Python
Pandas 同元素多列去重的实例
2018/07/03 Python
python 读取竖线分隔符的文本方法
2018/12/20 Python
Django后端接收嵌套Json数据及解析详解
2019/07/17 Python
PyCharm2019安装教程及其使用(图文教程)
2019/09/29 Python
Python使用QQ邮箱发送邮件实例与QQ邮箱设置详解
2020/02/18 Python
django 扩展user用户字段inlines方式
2020/03/30 Python
python 密码学示例——理解哈希(Hash)算法
2020/09/21 Python
Python创建简单的神经网络实例讲解
2021/01/04 Python
小米官方旗舰店:Xiaomi
2020/08/07 全球购物
软件缺陷的分类都有哪些
2014/08/22 面试题
博士学位自我鉴定范文
2013/12/26 职场文书
中秋节国旗下演讲稿
2014/09/05 职场文书
中学图书馆工作总结
2015/08/11 职场文书
Mysql关于数据库是否应该使用外键约束详解说明
2021/10/24 MySQL
Python列表的索引与切片
2022/04/07 Python
MySQL分区以及建索引的方法总结
2022/04/13 MySQL
MySQL安装失败的原因及解决步骤
2022/06/14 MySQL
使用python生成大量数据写入es数据库并查询操作(2)
2022/09/23 Python