我们每天都在敲代码,对着各种各样的类与继承。面向对象的编程设计方式,裹挟着PHP程序员加入 OOP 大军。
但是历史包袱太重,导致PHP的函数工具库的印象,那样的深刻。而诸如 Zend Symfony Laravel 框架出现,参与开发的人数众多,设计模式非常巧妙,让很多优秀的开发者把精力投入到 PHP 的 OOP 化。
那么问题来了,你写了这么多的 PHP 代码,搞得清楚 self 和 $this 的区别吗?本文就来简要说一说。
学习时间
简单地说,
- $this 指向当前的对象实例,self 指向当前类。
换句话说:
- $this->member 调用非静态属性/方法,self::number 调用静态属性/方法。
举个栗子,比较直观,说概念总是那么枯燥。
上面这个例子很明确地使用了静态属性,和动态属性,在构造函数内调用。实例化时执行。
如果反过来用就出错了,比如使用:
- self::$non_static_member . ' ' . $this->static_member;
self 调用了非静态属性,而 $this 调用了静态属性,这是错误的用法。
下面使用含有 $this 对象属性/方法调用重载了的函数方法,演示类的多态性:
上述文件执行后,返回值 Y::foo()。$this 作为实例化的 Y 对象,直接访问了其动态方法 foo()。而继承的 X 类的 foo() 方法被重写,并不执行。
现在换一种写法。
这次类 X 的 bar 方法我们使用 self::foo() 调用。那么很显然,self 就是 class X 本身,那么调用的也自然是 X 的 foo 方法。上述程序输出 X::foo()。
引申:尽量别用 self::,要用 static::
self 简单好用,但是作用域叫人恼火。因为它的作用域是在定义时的,而不是执行时的。比如这么个例子:
如果调用 Person::status(),返回 'Person is alive'。现在新建类,并继承该类:
执行 Deceased::status(),你期望得到什么,肯定是 Decased 类的 getStatus() 返回的值对么?可是结果返回了 Person::status() 的值。这是因为 status 方法被调用时,使用了 self::getStatus(),访问的是 Person 类的 getStatus() 方法。这是由 self 作用域决定的。
怎么才能返回期望的值?使用 static 替换 self。
还是那个规则:$this 引用的是当前的类的实例,static 引用的是当前的类本身。
我们从优秀的代码中学习用法。下面是 Laravel ValidationData 类内的一个静态方法。
写在最后
本文通过几个简单的小栗子,说明了 $this self static 之间的微妙差别,以及作用区间。希望大家在设计类的时候,能够谨慎地选用。