PHP中的self关键字详解


Posted in PHP onJune 23, 2019

前言

PHP群里有人询问self关键字的用法,答案是比较明显的:静态成员函数内不能用this调用非成员函数,但可以用self调用静态成员函数/变量/常量;其他成员函数可以用self调用静态成员函数以及非静态成员函数。随着讨论的深入,发现self并没有那么简单。鉴于此,本文先对几个关键字做对比和区分,再总结self的用法。

与parent、static以及this的区别

要想将彻底搞懂self,要与parent、static以及this区分开。以下分别做对比。

parent

self与parent的区分比较容易:parent引用父类/基类被隐盖的方法(或变量),self则引用自身方法(或变量)。例如构造函数中调用父类构造函数:

class Base {
 public function __construct() {
  echo "Base contructor!", PHP_EOL;
 }
}

class Child {
 public function __construct() {
  parent::__construct();
  echo "Child contructor!", PHP_EOL;
 }
}

new Child;
// 输出:
// Base contructor!
// Child contructor!

static

static常规用途是修饰函数或变量使其成为类函数和类变量,也可以修饰函数内变量延长其生命周期至整个应用程序的生命周期。但是其与self关联上是PHP 5.3以来引入的新用途:静态延迟绑定。

有了static的静态延迟绑定功能,可以在运行时动态确定归属的类。例如:

class Base {
 public function __construct() {
  echo "Base constructor!", PHP_EOL;
 }

 public static function getSelf() {
  return new self();
 }

 public static function getInstance() {
  return new static();
 }

 public function selfFoo() {
  return self::foo();
 }

 public function staticFoo() {
  return static::foo();
 }

 public function thisFoo() {
  return $this->foo();
 }

 public function foo() {
  echo "Base Foo!", PHP_EOL;
 }
}

class Child extends Base {
 public function __construct() {
  echo "Child constructor!", PHP_EOL;
 }

 public function foo() {
  echo "Child Foo!", PHP_EOL;
 }
}

$base = Child::getSelf();
$child = Child::getInstance();

$child->selfFoo();
$child->staticFoo();
$child->thisFoo();

程序输出结果如下:

Base constructor!
Child constructor!
Base Foo!
Child Foo!
Child Foo!

在函数引用上,self与static的区别是:对于静态成员函数,self指向代码当前类,static指向调用类;对于非静态成员函数,self抑制多态,指向当前类的成员函数,static等同于this,动态指向调用类的函数。

parent、self、static三个关键字联合在一起看挺有意思,分别指向父类、当前类、子类,有点“过去、现在、未来”的味道。

this

self与this是被讨论最多,也是最容易引起误用的组合。两者的主要区别如下:

  1. this不能用在静态成员函数中,self可以;
  2. 对静态成员函数/变量的访问,建议 用self,不要用$this::或$this->的形式;
  3. 对非静态成员变量的访问,不能用self,只能用this;
  4. this要在对象已经实例化的情况下使用,self没有此限制;
  5. 在非静态成员函数内使用,self抑制多态行为,引用当前类的函数;而this引用调用类的重写(override)函数(如果有的话)。

self的用途

看完与上述三个关键字的区别,self的用途是不是呼之即出?一句话总结,那就是:self总是指向“当前类(及类实例)”。详细说则是:

  1. 替代类名,引用当前类的静态成员变量和静态函数;
  2. 抑制多态行为,引用当前类的函数而非子类中覆盖的实现;

槽点

  1. 这几个关键字中,只有this要加$符号且必须加,强迫症表示很难受;
  2. 静态成员函数中不能通过$this->调用非静态成员函数,但是可以通过self::调用,且在调用函数中未使用$this->的情况下还能顺畅运行。此行为貌似在不同PHP版本中表现不同,在当前的7.3中ok;
  3. 在静态函数和非静态函数中输出self,猜猜结果是什么?都是string(4) "self",迷之输出;
  4. return $this instanceof static::class;会有语法错误,但是以下两种写法就正常:
$class = static::class;
return $this instanceof $class;
// 或者这样:
return $this instanceof static;

所以这是为什么啊?!

参考

When to use self over $this?

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对三水点靠木的支持。

PHP 相关文章推荐
非常好的php目录导航文件代码
Oct 09 PHP
Window下PHP三种运行方式图文详解
Jun 11 PHP
PHP上传图片进行等比缩放可增加水印功能
Jan 13 PHP
php表单请求获得数据求和示例
May 15 PHP
ThinkPHP中U方法的使用浅析
Jun 13 PHP
将酷狗krc歌词解析并转换为lrc歌词php源码
Jun 20 PHP
javascript some()函数用法详解
Nov 13 PHP
在WordPress中使用wp_count_posts函数来统计文章数量
Jan 05 PHP
示例详解Laravel的注册重构
Aug 14 PHP
PHP常用工具函数小结【移除XSS攻击、UTF8与GBK编码转换等】
Apr 27 PHP
PHP时间相关常用函数用法示例
Jun 03 PHP
PHP之header函数详解
Mar 02 PHP
php面向对象程序设计入门教程
Jun 22 #PHP
PHP字符串中抽取子串操作实例分析
Jun 22 #PHP
微信公众号之主动给用户发送消息功能
Jun 22 #PHP
Yii 使用intervention/image拓展实现图像处理功能
Jun 22 #PHP
PHP实现给定一列字符,生成指定长度的所有可能组合示例
Jun 22 #PHP
PHP中strtr与str_replace函数运行性能简单测试示例
Jun 22 #PHP
PHP判断函数是否被定义的方法
Jun 21 #PHP
You might like
咖啡界又出新概念,无需咖啡豆的分子咖啡
2021/03/03 咖啡文化
PHP中的正规表达式(二)
2006/10/09 PHP
laravel框架中表单请求类型和CSRF防护实例分析
2019/11/23 PHP
Javascript实例教程(19) 使用HoTMetal(3)
2006/12/23 Javascript
JavaScript开发时的五个注意事项
2007/12/08 Javascript
ext form 表单提交数据的方法小结
2008/08/08 Javascript
JavaScript调用堆栈及setTimeout使用方法深入剖析
2013/02/16 Javascript
jquery easyui中treegrid用法的简单实例
2014/02/18 Javascript
Javascript编写俄罗斯方块思路及实例
2015/07/07 Javascript
Javascript使用function创建类的两种方法(推荐)
2016/11/19 Javascript
详解http访问解析流程原理
2017/10/18 Javascript
详解vue文件中使用echarts.js的两种方式
2018/10/18 Javascript
antd 表格列宽自适应方法以及错误处理操作
2020/10/27 Javascript
[03:55]2014DOTA2国际邀请赛 Fnatic经理采访赢DK在情理之中
2014/07/10 DOTA
[01:11:21]DOTA2-DPC中国联赛 正赛 Phoenix vs CDEC BO3 第三场 3月7日
2021/03/11 DOTA
python client使用http post 到server端的代码
2013/02/10 Python
在Python中操作字符串之startswith()方法的使用
2015/05/20 Python
Python易忽视知识点小结
2015/05/25 Python
Python实现数通设备端口使用情况监控实例
2015/07/15 Python
Python list操作用法总结
2015/11/10 Python
Python 高级专用类方法的实例详解
2017/09/11 Python
python3实现逐字输出的方法
2019/01/23 Python
numpy下的flatten()函数用法详解
2019/05/27 Python
Python实现时间序列可视化的方法
2019/08/06 Python
浅谈在django中使用filter()(即对QuerySet操作)时踩的坑
2020/03/31 Python
html5音频_动力节点Java学院整理
2018/08/22 HTML / CSS
泰国王权免税店官方网站:KingPower
2019/03/11 全球购物
js正则匹配markdown里的图片标签的实现
2021/03/24 Javascript
五年级英语教学反思
2014/01/31 职场文书
优秀团干部个人事迹
2014/05/29 职场文书
学校学习雷锋活动总结
2014/07/03 职场文书
社区党的群众路线教育实践活动领导班子对照检查材料
2014/09/25 职场文书
征用土地赔偿协议书
2014/09/26 职场文书
乡镇干部个人整改措施思想汇报
2014/10/10 职场文书
2014年煤矿工人工作总结
2014/12/08 职场文书
Android中View.post和Handler.post的关系
2022/06/05 Java/Android