分享web开发知识

注册/登录|最近发布|今日推荐

主页 IT知识网页技术软件开发前端开发代码编程运营维护技术分享教程案例
当前位置:首页 > 教程案例

PHP常见概念混淆(七)之self、static、parent的区别

发布时间:2023-09-06 02:08责任编辑:蔡小小关键词:PHP

前言

首先,这个 static 符号跟 static(静态)关键字不是一个东西。这三个符号在PHP对象中共有两种用法:

  1. 在类内部,可以使用 new selfnew staticnew parent 创建新对象
  2. 可以使用 self::static::parent::调用静态变量和方法。

创建新对象

 <?php class test{ ???public static function test_self(){ ???????return new self(); ???} ???public static function test_static(){ ???????return new static(); ???} ???public static function test_parent(){ ???????return new parent(); ???}}class test2 extends test{ ???public static function test_parent(){ ???????return new parent(); ???}}class test3 extends test2{}echo get_class(test::test_self()); ?//testecho get_class(test::test_static()); //testecho get_class(test2::test_self()); ?//testecho get_class(test2::test_static()); //test2echo get_class(test2::test_parent()); //testecho get_class(test3::test_self()); //testecho get_class(test3::test_static()); //test3 echo get_class(test3::test_parent()); //test

由以上这个例子可以得出:

  1. new self 创建的对象是定义 new self的类创建的对象
  2. new static 创建的对象是执行 new static的类创建的对象
  3. new parent 创建的对象是定义 new parent 的父类创建的对象(PHP5.3引进)

调用静态变量

概念
  1. 转发调用(forwarding call):所谓的"转发调用"指的是通过以下几种方式进行的静态调用:self::parent::static:: 以及 forward_static_call().即在进行静态调用时未指名类名的调用属于转发调用。
  2. 非转发调度(non-forwarding call):非转发调用其实就是明确指定类名的静态调用(foo::bar())和非静态调用($foo->bar())。即明确地指定类名的静态调用和非静态调用。
  3. 后期静态绑定(Late Static Bindings ):"后期静态绑定"的意思是说,static:: 不再被解析为定义当前方法所在的类,而是在实际运行时计算的。
不存在继承的时候

self和static无区别。

  • 在静态函数中,self和static可以调用静态属性和静态方法(沒有实例化类,因此不能呼叫非静态的属性和方法)。
  • 在非静态函数中, self和static可以调用非静态属性和非静态方法。

     <?php ?class Demo{ ??????????public static $static; ??????????public $Nostatic; ???????????public function __construct(){ ??????????????????self::$static = "static"; ??????????????????$this->Nostatic = "Nostatic"; ?????} ?????public static function get(){ ?????????return __CLASS__; ?????} ?????public function show(){ ?????????return "this is function show with ".$this->Nostatic; ?????} ????public function test(){ ????????echo Demo::$static."\n"; ?//使用类型调用静态属性 ????static ????????echo Demo::get()."\n"; ?//使用类名调用非静态方法 ???Demo ????????echo Demo::show()."\n"; ?//使用类名调用静态方法 ??this is function show with Nostatic ????????echo self::$static."\n"; ?//self调用静态属性 ??????static ????????echo self::get()."\n"; ?//self调用非静态方法 ?????Demo ????????echo self::show()."\n"; ??//self调用静态方法 ??????this is function show with Nostatic ????????echo static::$static."\n";//static调用静态属性 ????static ????????echo static::get()."\n";//static调用非静态方法 ???Demo ????????echo static::show()."\n"; //static调用静态方法 ??this is function show with Nostatic ?????} ?} $obj = new Demo(); $obj->test();
存在继承关系的时候
  • self调用的方法和属性始终表示当前类的方法和属性
  • static调用的方法和属性为当前执行的类的方法和属性
  • parent调用的方法和属性为父类的方法和属性

     ?<?php ???class A{ ????static $test = "AAA"; ????static function getClassName(){ ?????????return "A"; ?????} ?????function getClassName2(){ ?????????return "AA"; ?????} ?????static function testSelf(){ ?????????echo self::getClassName(); ?????????echo "\n"; ?????????echo self::getClassName2(); ???????????echo "\n"; ?????????echo self::$test; ?????} ????static function testStatic(){ ?????????echo static::getClassName(); ?????????echo "\n"; ?????????echo static::getClassName2(); ?????????echo "\n"; ?????????echo static::$test; ?????} ?} ?class B extends A{ ??????static $test = "BBB"; ??????static function getClassName(){ ???????????return "B"; ???????} ??????function getClassName2(){ ???????????return "BB"; ???????} ??????static function testParent(){ ???????????echo parent::getClassName(); ???????????echo "\n"; ???????????echo parent::getClassName2(); ???????????echo "\n"; ???????????echo parent::$test; ?????} ?} ??class C extends B{ ???????static $test = "CCC"; ???????static function getClassName(){ ???????????return "C"; ???????} ???????function getClassName2(){ ???????????return "CC"; ???????} ???????static function testParent(){ ????????????echo parent::getClassName(); ????????????echo "\n"; ????????????echo parent::getClassName2(); ????????????echo "\n"; ????????????echo parent::$test; ???????} ?} ?B::testSelf(); // A AA AAA ?echo "\n"; ?B::testStatic();// B BB BBB ?echo "\n"; ?B::testParent();// A AA AAA ?echo "\n"; ?C::testSelf(); // A AA AAA ?echo "\n"; ?C::testStatic(); // C CC CCC ?echo "\n"; ?C::testParent(); // B BB BBB

解析一下手册里的例子

<?php class A { ???????public static function foo() { ????????????????static::who(); ???????} ???????public static function who() { ???????????????echo __CLASS__."\n"; ???????} } class B extends A { ???????public static function test() { ???????????????A::foo(); ???????????????parent::foo(); ???????????????self::foo(); ???????} ???????public static function who() { ????????????????echo __CLASS__."\n"; ????????} } class C extends B { ????????public static function who() { ???????????????echo __CLASS__."\n"; ????????}} C::test(); ?>

最后输出为 A C C
单独拿出进行分析

public static function test() { ???????????A::foo(); ???????????parent::foo(); ???????????self::foo(); ????} ????
  1. A::foo():非转发请求,直接调用A的foo()方法,在任何地方调用结果都是一样的
  2. parent::foo():在B类中写着,调用B的父类A的方法foo()parent的用法);A类中的foo()中执行 static::who(),寻找上一个非转发请求的类名(在A类的foo()方法中写上get_called_class(),可得为C,由此可知就是当前执行的类),所以调用C类的who()方法(这一步就可以理解为后期静态绑定,即为static的用法);C类中重写了who()方法,所以结果为C;如果去掉C类中的who()方法,会调用B类中的who()方法;如果再去掉B类中的who()方法,会调用A类中的who()方法。
  3. self::foo():执行B类中的foo()方法(self的用法),B类中没有foo()方法,于是继承了A类的foo()方法,如果B类中定义了foo()方法,则执行B类中的foo()方法;执行A类的foo()方法,如上.

PHP常见概念混淆(七)之self、static、parent的区别

原文地址:https://www.cnblogs.com/qiye5757/p/9437971.html

知识推荐

我的编程学习网——分享web前端后端开发技术知识。 垃圾信息处理邮箱 tousu563@163.com 网站地图
icp备案号 闽ICP备2023006418号-8 不良信息举报平台 互联网安全管理备案 Copyright 2023 www.wodecom.cn All Rights Reserved