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
深入解析php之apc
May 15 PHP
php利用反射实现插件机制的方法
Mar 14 PHP
php数据访问之查询关键字
May 09 PHP
一段实用的php验证码函数
May 19 PHP
thinkphp多表查询两表有重复相同字段的完美解决方法
Sep 22 PHP
PHP获取当前执行php文件名的代码
Mar 02 PHP
TP(thinkPHP)框架多层控制器和多级控制器的使用示例
Jun 13 PHP
让Laravel API永远返回JSON格式响应的方法示例
Sep 05 PHP
Laravel框架实现的使用smtp发送邮件功能示例
Mar 12 PHP
Laravel如何创建服务器提供者实例代码
Apr 15 PHP
laravel config文件配置全局变量的例子
Oct 13 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
Cakephp 执行主要流程
2010/03/24 PHP
PHP错误WARNING: SESSION_START() [FUNCTION.SESSION-START]解决方法
2014/05/04 PHP
php结合js实现点击超链接执行删除确认操作
2014/10/31 PHP
php+ajax注册实时验证功能
2016/07/20 PHP
php、java、android、ios通用的3des方法(推荐)
2016/09/09 PHP
身份证号码前六位所代表的省,市,区, 以及地区编码下载
2007/04/12 Javascript
JS Timing
2007/04/21 Javascript
javascript mouseover、mouseout停止事件冒泡的解决方案
2009/04/07 Javascript
Javascript Math ceil()、floor()、round()三个函数的区别
2010/03/09 Javascript
简单的jquery拖拽排序效果实现代码
2011/09/20 Javascript
JS 退出系统并跳转到登录界面的实现代码
2013/06/29 Javascript
jQuery表格排序组件-tablesorter使用示例
2014/05/26 Javascript
基于javascript的JSON格式页面展示美化方法
2014/07/02 Javascript
JS实现的车标图片提示效果代码
2015/10/10 Javascript
jQuery zTree 异步加载添加子节点重复问题
2017/11/29 jQuery
js中this的指向问题归纳总结
2018/11/28 Javascript
详解@angular/cli 改变默认启动端口两种方式
2018/11/29 Javascript
vue中created和mounted的区别浅析
2019/08/13 Javascript
Vue防止白屏添加首屏动画的实例
2019/10/31 Javascript
Vue单页面应用中实现Markdown渲染
2021/02/14 Vue.js
Python3.x和Python2.x的区别介绍
2013/02/12 Python
Python实现读取及写入csv文件的方法示例
2018/01/12 Python
详解python中的线程与线程池
2019/05/10 Python
python使用pygame模块实现坦克大战游戏
2020/03/25 Python
Python何时应该使用Lambda函数
2019/07/02 Python
Django ModelForm组件使用方法详解
2019/07/23 Python
TensorFlow MNIST手写数据集的实现方法
2020/02/05 Python
使用HTML和CSS3绘制基本卡通图案的示例分享
2015/11/06 HTML / CSS
用React加CSS3实现微信拆红包动画效果
2017/03/13 HTML / CSS
北京某公司的.net笔试题
2014/03/20 面试题
各营销点岗位职责范本
2014/03/05 职场文书
优秀的2014年两会精神解读
2014/03/17 职场文书
2014年个人总结范文
2015/03/09 职场文书
女儿满月酒致辞
2015/07/29 职场文书
厉行节约工作总结
2015/08/12 职场文书
导游词之日本富士山
2020/01/06 职场文书