<?php
class A {
const CONST_NAME = 'A';
public static $static_name = 'A';
public static $static_name_no_rewrite = 'A'; private static $_instance; public static function static_func() {
echo 'A';
} public static function static_func_no_rewrite() {
echo 'A';
} public static function static_extends_self() {
echo self::$static_name;
} public static function static_extends_no_rewrite_self() {
echo self::$static_name;
} public static function static_extends_no_vars() {
return self::$static_no_vars;
} public static function get_instance() {
if (empty(self::$_instance)) {
self::$_instance = new self();
}
return self::$_instance;
}
} class B extends A {
const CONST_NAME = 'B';
public static $static_name = 'B';
public static $static_no_vars = 'B'; //重写
public static function static_func() {
echo 'B';
} public static function static_extends_self() {
echo self::$static_name;
}
}
//是否继承后是有了一个父类的引用 //1.查看父类和子类的初始情况
$str = '1.查看父类和子类的初始情况';
echo '<hr />';
echo $str, '<br />';
echo 'A::CONST_NAME->', A::CONST_NAME, '<br />';
echo 'B::CONST_NAME->', B::CONST_NAME, '<br />';
echo 'A::$static_name->', A::$static_name, '<br />';
echo 'B::$static_name->', B::$static_name, '<br />';
echo 'B::$static_name_no_rewrite->', B::$static_name_no_rewrite, '<br />';
echo 'B::$static_name_no_rewrite->', B::$static_name_no_rewrite, '<br />';
echo '结论:静态属性或类常量可以被继承<br />';
echo '<hr />'; //2.测试重定义的类属性 修改子类(父类)的值父类(子类)的值是否会改变
$str = '2.测试重定义的类属性 修改子类(父类)的值父类(子类)的值是否会改变';
echo $str, '<br />';
A::$static_name = 'M_A';
echo "修改->A::\$static_name = 'M_A'<br />";
echo 'A::$static_name->', A::$static_name, '<br />';
echo 'B::$static_name->', B::$static_name, '<br />';
B::$static_name = 'M_B';
echo "修改->B::\$static_name = 'M_B'<br />";
echo 'A::$static_name->', A::$static_name, '<br />';
echo 'B::$static_name->', B::$static_name, '<br />';
echo '结论:继承关系的两个类,如果子类重定义了父类的静态属性,则修改父类(子类)的静态属性或常量不会影响到子类(父类)的对应静态属性<br />';
echo '<hr />'; //3.测试没有重定义的类属性 修改子类(父类)的值父类(子类)的值是否会改变
$str = '3.测试没有重定义的类属性 修改子类(父类)的值父类(子类)的值是否会改变';
echo $str, '<br />';
echo 'A::$static_name_no_rewrite->', A::$static_name_no_rewrite, '<br />';
echo 'B::$static_name_no_rewrite->', B::$static_name_no_rewrite, '<br />';
A::$static_name_no_rewrite = 'M_A';
echo "修改->A::\$static_name_no_rewrite = 'M_A'<br />";
echo 'A::$static_name_no_rewrite->', A::$static_name_no_rewrite, '<br />';
echo 'B::$static_name_no_rewrite->', B::$static_name_no_rewrite, '<br />';
B::$static_name_no_rewrite = 'M_B';
echo "修改->B::\$static_name_no_rewrite = 'M_B'<br />";
echo 'A::$static_name_no_rewrite->', A::$static_name_no_rewrite, '<br />';
echo 'B::$static_name_no_rewrite->', B::$static_name_no_rewrite, '<br />';
echo '结论:如果子类没有重定义父类的某个静态属性,则修改父类(子类)则子类(父类)的对应静态属性也随之改变<br />';
echo '更深结论:子类继承父类会有一个父类的引用指向父类那块内存,如果子类没有重定义一个属性(放到本类内存下)就会调用父类的属性<br />';
echo '<hr />'; //4.静态方法是否能继承
$str = '4.静态方法是否能继承';
echo $str, '<br />';
echo 'A::static_func_no_rewrite()->', A::static_func_no_rewrite(), '<br />';
echo 'B::static_func_no_rewrite()->', B::static_func_no_rewrite(), '<br />';
echo '结论:静态方法可以继承!';
echo '<hr />'; //5.静态方法是否能被重写
$str = '5.静态方法是否能被重写';
echo $str, '<br />';
echo 'A::static_func()->', A::static_func(), '<br />';
echo 'B::static_func()->', B::static_func(), '<br />';
echo '结论:静态方法可以被重写!';
echo '<hr />'; //6.父类静态方法中有self, 子类继承此方法没有重写后调用的self是谁?
$str = '6.父类静态方法中有self, 子类继承此方法没有重写后调用的self是谁?';
echo $str, '<br />';
echo 'A::static_extends_no_rewrite_self()->', A::static_extends_no_rewrite_self(), '<br />';
echo 'B::static_extends_no_rewrite_self()->', B::static_extends_no_rewrite_self(), '<br />';
echo '结论:与3相同, 如若没有重写, 则只是子类对父类的一个引用, 如果子类中没有, 就去父类中寻找, 而且找到的也是依赖于父类!';
echo '更深结论:在PHP单例的继承中, 父类::get_instance()方法中返回new self(), 如果子类没有重写, 则返回父类的实例!';
echo '<hr />'; //7.父类静态方法中有self, 子类继承此方法重写后调用的self是谁?
$str = '7.父类静态方法中有self, 子类继承此方法重写后调用的self是谁?';
echo $str, '<br />';
echo 'A::static_extends_self()->', A::static_extends_self(), '<br />';
echo 'B::static_extends_self()->', B::static_extends_self(), '<br />';
echo '结论:子类覆盖父类的方法后, 如果其中有self属性, 则此时会覆盖父类的self, 即此时的self为对象本身';
echo '<hr />'; //8.父类静态方法调用本身没有子类却有的静态属性(来验证上述6推测, 如果是引用, 则会为空)
$str = '8.子类调用父类静态方法调用父类没有子类却有的self的静态属性(来验证上述6推测, 如果是引用, 则会为空)';
echo $str, '<br />';
//echo 'B::static_extends_no_vars()->', B::static_extends_no_vars(), '<br />';
echo 'B::static_extends_no_vars()->报错<br />';
echo '结论:程序报错, 显示A没有此属性, 说明是子类对父类的一个引用!子类如果没有重写父类的属性或静态方法, 则调用父类的!包括self!';
echo '<hr />'; //9.测试下6中的推断, 即单例模式中self
$str = '9.测试下6中的推断, 即单例模式中self';
echo $str, '<br />';
echo 'get_class(B::get_instance())->', get_class(B::get_instance()), '<br />';
echo '结论:说明6中的更深推断是正确的!';
echo '<hr />'; //10.测试实例调用静态属性或常量
$str = '测试实例调用静态属性或常量';
echo $str, '<br />';
$a = new A();
//echo '$a = new A();echo $a->CONST_NAME->', $a->CONST_NAME, '<br />';
echo '$a = new A();echo $a->CONST_NAME->报错<br />';
//echo 'echo $a->$static_name->', $a->$static_name, '<br />';
echo 'echo $a->$static_name->报错<br />';
echo '结论:实例不能调用类的静态属性或常量!调用静态属性会转变为变量的变量, 调用常量会调用实例的对应属性!';
echo '<hr />'; //11.测试实例调用静态方法
$str = '测试实例调用静态属性或常量';
echo $str, '<br />';
echo '$a->static_func()->', $a->static_func(), '<br />';
echo '$a->static_extends_self()->', $a->static_extends_self(), '<br />';
echo '结论:实例可以调用静态方法, 甚至静态方法中包含self!';
echo '<hr />';

PHP-静态方法(static)继承等分析的更多相关文章

  1. 第32节:Java中-构造函数,静态方法,继承,封装,多态,包

    构造函数实例 class Cat{ // 设置私有的属性 name private String name; // 设置name的方法 public void setName(String Name) ...

  2. .net类中静态方法的继承

    父类中的静态方法,继承的子类能不能调用?一直在这里有疑惑,即使在下面的测试之后,也只是得到了结论,不明原理. class ClsParent { public static void ShowSth( ...

  3. static 继承

    静态方法大家应该都比较熟悉,在这里主要谈一下静态方法在继承时的一些注意事项. 1.父类方法如果是静态方法,子类不能覆盖为非静态方法: 2.父类方法如果是非静态方法,子类不能覆盖为静态方法: 3.父类静 ...

  4. koa 基础(十八)es6中的类、静态方法、继承

    1.app.js /** * es6中的类.静态方法.继承 */ // 定义Person类 class Person { constructor(name, age) { /*类的构造函数,实例化的时 ...

  5. 关于static继承的问题

    c++primer 15.2.7节关于static继承的意思是,父类和子类共享static函数或者static成员变量,并且子类要访问还要受它们的权限限制,下面是看到的另一个例子 class Base ...

  6. java中静态变量与静态方法的继承(转)

    总结: 1.静态变量与静态方法说继承并不确切,静态方法与变量是属于类的方法与变量.而子类也属于超类,比如说Manage extends Employee,则Manage也是一个Employee,所以子 ...

  7. Python中的单继承与多继承实例分析

    Python中的单继承与多继承实例分析 本文实例讲述了Python中的单继承与多继承.分享给大家供大家参考,具体如下: 单继承 一.介绍 Python 同样支持类的继承,如果一种语言不支持继承,类就没 ...

  8. koa 基础(十七)原生 JS 中的类、静态方法、继承

    1.app.js /** * 原生 JS 中的类.静态方法.继承 * es5中的类和静态方法 */ function Person(name, age) { // 构造函数里面的方法和属性 this. ...

  9. java中静态属性和和静态方法的继承问题 以及多态的实质

    首先结论是:java中静态属性和和静态方法可以被继承,但是没有被重写(overwrite)而是被隐藏. 静态方法和属性是属于类的,调用的时候直接通过类名.方法名完成的,不需继承机制就可以调用如果子类里 ...

随机推荐

  1. Google Code Jam 2010 Round 1C Problem A. Rope Intranet

    Google Code Jam 2010 Round 1C Problem A. Rope Intranet https://code.google.com/codejam/contest/61910 ...

  2. javascript:window.history.forward(1);

    javascript:window.history.forward(1);[转] 接下来我们要讨论的方法以后退按钮本身为中心,而不是浏览器缓存.这儿有一篇文章Rewiring the Back But ...

  3. 解决Visual Studio 2015创建工程时的“DNX SDK version 'dnx-clr-win-x86.1.0.0-beta5' failed to install.”错误

    前段时间发布了Visual Studio2015,在后,发现创建Asp.Net工程和时,出现了"DNX SDK version 'dnx-clr-win-x86.1.0.0-beta5' f ...

  4. STM32 USB VBUS 监控

    OTG_FS general core configuration register (OTG_FS_GCCFG) Bit 21 NOVBUSSENS: VBUS sensing disable op ...

  5. Spring bean配置继承

    在 Spring,继承是用为支持bean设置一个 bean 来分享共同的值,属性或配置. 一个子 bean 或继承的bean可以继承其父 bean 的配置,属性和一些属性.另外,子 Bean 允许覆盖 ...

  6. SQL Server的thread scheduling(线程调度)

      https://www.zhihu.com/question/53125711/answer/134461670 https://www.zhihu.com/question/53125711   ...

  7. golang(一)-for 循环

    golang 的循环控制中大多还是和java 很相似的 , 不过golang只有一种循环 就是for循环: for 有三个循环控制关键字 : break . continue . goto  其中   ...

  8. Oracle初始安装内存设置参考

      预备知识 shared memory:共享内存段: 一个内存区域,可以被不同的进程读取.oracle使用它来构成sga.oracle使用以下三种方法来创建一个sga区: 1. 使用单个共享内存段. ...

  9. vc维的解释

    在做svm的时候我们碰到了结构风险最小化的问题,结构风险等于经验风险+vc置信范围,当中的vc置信范围又跟样本的数量和模型的vc维有关,所以我们看一下什么是vc维 首先看一下vc维的定义:对一个指标函 ...

  10. 2)Linux程序设计入门--进程介绍

    )Linux程序设计入门--进程介绍 Linux下进程的创建 前言: 这篇文章是用来介绍在Linux下和进程相关的各个概念.我们将会学到: 进程的概念 进程的身份 进程的创建 守护进程的创建 .进程的 ...