PHP程序员:你过来,给我说说 $this,self,static 有什么区别?

程序员之怒 2020-06-23

引言

我们每天都在敲代码,对着各种各样的类与继承。面向对象的编程设计方式,裹挟着PHP程序员加入 OOP 大军。

PHP程序员:你过来,给我说说 $this,self,static 有什么区别?

但是历史包袱太重,导致PHP的函数工具库的印象,那样的深刻。而诸如 Zend Symfony Laravel 框架出现,参与开发的人数众多,设计模式非常巧妙,让很多优秀的开发者把精力投入到 PHP 的 OOP 化。

那么问题来了,你写了这么多的 PHP 代码,搞得清楚 self 和 $this 的区别吗?本文就来简要说一说。

学习时间

简单地说,

$this 指向当前的对象实例,self 指向当前类。 

换句话说:

$this->member 调用非静态属性/方法,self::number 调用静态属性/方法。 

举个栗子,比较直观,说概念总是那么枯燥。

PHP程序员:你过来,给我说说 $this,self,static 有什么区别?

上面这个例子很明确地使用了静态属性,和动态属性,在构造函数内调用。实例化时执行。

如果反过来用就出错了,比如使用:

self::$non_static_member . ' ' . $this->static_member; 

self 调用了非静态属性,而 $this 调用了静态属性,这是错误的用法。

下面使用含有 $this 对象属性/方法调用重载了的函数方法,演示类的多态性:

PHP程序员:你过来,给我说说 $this,self,static 有什么区别?

上述文件执行后,返回值 Y::foo()。$this 作为实例化的 Y 对象,直接访问了其动态方法 foo()。而继承的 X 类的 foo() 方法被重写,并不执行。

现在换一种写法。

PHP程序员:你过来,给我说说 $this,self,static 有什么区别?

这次类 X 的 bar 方法我们使用 self::foo() 调用。那么很显然,self 就是 class X 本身,那么调用的也自然是 X 的 foo 方法。上述程序输出 X::foo()。

引申:尽量别用 self::,要用 static::

self 简单好用,但是作用域叫人恼火。因为它的作用域是在定义时的,而不是执行时的。比如这么个例子:

PHP程序员:你过来,给我说说 $this,self,static 有什么区别?

如果调用 Person::status(),返回 'Person is alive'。现在新建类,并继承该类:

PHP程序员:你过来,给我说说 $this,self,static 有什么区别?

执行 Deceased::status(),你期望得到什么,肯定是 Decased 类的 getStatus() 返回的值对么?可是结果返回了 Person::status() 的值。这是因为 status 方法被调用时,使用了 self::getStatus(),访问的是 Person 类的 getStatus() 方法。这是由 self 作用域决定的。

怎么才能返回期望的值?使用 static 替换 self。

还是那个规则:$this 引用的是当前的类的实例,static 引用的是当前的类本身。

我们从优秀的代码中学习用法。下面是 Laravel ValidationData 类内的一个静态方法。

PHP程序员:你过来,给我说说 $this,self,static 有什么区别?

相关推荐