PHP 的 self 关键字用法
之前有人询问
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
是被讨论最多,也是最容易引起误用的组合。两者的主要区别如下:
this
不能用在静态成员函数中,self
可以;- 对静态成员函数/变量的访问, 建议 用
self
,不要用$this::
或$this->
的形式; - 对非静态成员变量的访问,不能用
self
,只能用this
; this
要在对象已经实例化的情况下使用,self
没有此限制;- 在非静态成员函数内使用,
self
抑制多态行为,引用当前类的函数;而this
引用调用类的重写(override)函数(如果有的话)。
self
的用途
看完与上述三个关键字的区别, self
的用途是不是呼之即出?一句话总结,那就是: self
总是指向“当前类(及类实例)”。详细说则是:
- 替代类名,引用当前类的静态成员变量和静态函数;
- 抑制多态行为,引用当前类的函数而非子类中覆盖的实现;
槽点
- 这几个关键字中,只有
this
要加$
符号且必须加,强迫症表示很难受; - 静态成员函数中不能通过
$this->
调用非静态成员函数,但是可以通过self::
调用,且在调用函数中未使用$this->
的情况下还能顺畅运行。此行为貌似在不同PHP版本中表现不同,在当前的7.3中ok; - 在静态函数和非静态函数中输出
self
,猜猜结果是什么?都是string(4) "self"
,迷之输出; return $this instanceof static::class;
会有语法错误,但是以下两种写法就正常:
$class = static::class;
return $this instanceof $class;
// 或者这样:
return $this instanceof static;
很多PHPer在进阶的时候总会遇到一些问题和瓶颈,业务代码写多了没有方向感,不知道该从那里入手去提升,对此我整理了一些资料,包括但不限于:分布式架构、高可扩展、高性能、高并发、服务器性能调优、TP6,laravel,YII2,Redis,Swoole、Swoft、Kafka、Mysql优化、shell脚本、Docker、微服务、Nginx等多个知识点高级进阶干货需要的可以免费分享给大家,需要的加群(点击→)677079770
PHP 的 self 关键字用法的更多相关文章
- ava下static关键字用法详解
Java下static关键字用法详解 本文章介绍了java下static关键字的用法,大部分内容摘自原作者,在此学习并分享给大家. Static关键字可以修饰什么? 从以下测试可以看出, static ...
- 巨人大哥谈Java中的Synchronized关键字用法
巨人大哥谈Java中的Synchronized关键字用法 认识synchronized 对于写多线程程序的人来说,经常碰到的就是并发问题,对于容易出现并发问题的地方价格synchronized基本上就 ...
- synchronized关键字用法
看到网上很多讲synchronized关键字用法的文章,说的都很有道理,也很深刻,但是看完总感觉脑袋里还是有点乱乱的.经过一番自己的思考后,想从自己的思考角度出发,来说一说synchronized关键 ...
- Java-Runoob-高级教程-实例-方法:09. Java 实例 – continue 关键字用法-un
ylbtech-Java-Runoob-高级教程-实例-方法:09. Java 实例 – continue 关键字用法 1.返回顶部 1. Java 实例 - continue 关键字用法 Java ...
- Java-Runoob-高级教程-实例-方法:08. Java 实例 – break 关键字用法
ylbtech-Java-Runoob-高级教程-实例-方法:08. Java 实例 – break 关键字用法 1.返回顶部 1. Java 实例 - break 关键字用法 Java 实例 Ja ...
- Java-Runoob-高级教程-实例-方法:07. Java 实例 – instanceOf 关键字用法
ylbtech-Java-Runoob-高级教程-实例-方法:07. Java 实例 – instanceOf 关键字用法 1.返回顶部 1. Java 实例 - instanceof 关键字用法 ...
- C#中var关键字用法分析
原文连接 本文实例分析了C#中var关键字用法.分享给大家供大家参考.具体方法如下: C#关键字是伴随着.NET 3.5以后,伴随着匿名函数.LINQ而来, 由编译器帮我们推断具体的类型.总体来说,当 ...
- sql server if exists和 if not exists 的关键字用法
if exists和if not exists关键字用法 1.介绍 if not exists 即如果不存在,if exists 即如果存在 2.使用 a.判断数据库不存在时 if not ...
- Java 实例 - instanceof 关键字用法
Java 实例 - instanceof 关键字用法 instanceof 是 Java 的一个二元操作符,类似于 ==,>,< 等操作符. instanceof 是 Java 的保留关键 ...
- C#中default 、base 、this关键字用法简介
C#中default关键字用法简介 default 关键字可在switch语句或泛型代码中使用.switch语句:指定默认标签.泛型代码:指定类型参数的默认值.对于引用类型为空,对于值类型为零swi ...
随机推荐
- WebShell代码分析溯源(一)
WebShell代码分析溯源(一) 一.一句话变形马样本 <?php $_GET['POST']($_POST['GET']);?> 二.代码分析 1.调整代码格式 <?php $_ ...
- js数据交互——fetch
什么是fetch? Fetch被称为下一代Ajax技术,采用Promise方式来处理数据. 是一种简洁明了的API,比XMLHttpRequest更加简单易用.fetch是原生的(无需引入任何库和框架 ...
- C# 8 - using声明 和 异步流
这两个主题没什么关系,但是怕文章太短被移除主页. using声明 using语句块 尽管.NET Core运行时有垃圾收集器(GC)来负责内存清理工作,但是我们还是要自己确保当非托管资源不再使用的时候 ...
- JVM(4) 类文件结构
一.实现“平台无关性” 字节码(ByteCode)存储格式和虚拟机是实现语言无关性的基础.Java虚拟机不和包括Java在内的任何语言绑定,它只与“Clas”文件这种特定的二进制文件格式所关联,Cla ...
- Docker安装ElasticSearch 以及使用LogStash实现索引库和数据库同步
1:下载 ElasticSearch 镜像 docker pull docker.io/elasticsearch:5.6.8 2:创建 ElasticSearch 容器: 注意:5.0默认分配jvm ...
- 第三十八章 POSIX线程(二)
线程属性 初始化与销毁属性 int pthread_attr_init(pthread_attr_t *attr); int pthread_attr_destroy(pthread_attr_t * ...
- 使用“反向传播”迭代法求解y=√10
X=√10,求X,也就是求Y=10 =X2 , X是多少. *重要的思想是,如何转化为可迭代求解的算法问题. *解数学问题,第一时间画图,求导,“直线化”. Y = X2 假如已知Y = 10 ,要求 ...
- MIT线性代数:15.子空间的投影
- 前端与算法 leetcode 1. 两数之和
目录 # 前端与算法 leetcode 1. 两数之和 题目描述 概要 提示 解析 解法一:暴力法 解法二:HashMap法 算法 传入[1, 2], [11, 1, 2, 3, 2]的运行结果 执行 ...
- 苹果审核ipv6海外解决思路-About APP Store
原始简书文章地址(也是我自己的) 首先声明,一我不负责涉及你们内部服务器. 二是好好读文章,别人能过,你们也能过 苹果6月1日出的IPV6协议阻碍了国内大多数积极开发者,我司也不外乎,经过三次被拒后, ...