谈谈PHP继承的那些事,我的个人见解

 网上经常流传出php是语言鄙视链最低端的那个,曾经大学学java,毕设用java,刚出来培训用java的我,在最初工作的2、3年时对php的面向对象也是颇有意见,总觉得【不伦不类】,更别提对js的看法了。但是这些观点都在经历越来越多的项目之后逐渐的淡化,甚至改观。这里面包含着自己对项目、技术有着更多的理解,同时,在这些年里,Web环境、技术也在不停的更新。不过今天不是来聊这些东西的,对于以上的问题,我的观点可以总结为:技术是工具、手段,不合适就升级、换,就这么简单。

  话归原题。虽然写php已经是将近8年的功底了,但因为工作关系,经常需要涉及前后端的各种代码,容易精分,也总会记岔。最近发生的一件事情让我觉得,或许写下来能够让自己清醒一点。

  在某一年写某个模块时用到了static成员,在实现子类的过程中发现他们也共享着父类这个成员的值,具体来说就是我在某个子类A中改变了那个成员值,在另外一个子类B使用的时候结果意外的得到了A覆盖后的值。当时以为,原来static成员是在从声明的地方开始的整个类别树中共享的。后来一直隐约记得这个结论,在平常的代码里面更谨慎的使用static成员,除非确认写的类是个独立的工具类,不然不轻易使用static。

  直到有一天我的老大跟我商量升级我之前写的一个BaseModel,他无意中问我:好像你不喜欢用static成员?我说没有啊,因为考虑到BaseModel会被经常继承成各种Model,如果我在这里用了static的话,将来容易踩坑。他表示不理解,然后过来与我辩论。我很义正言辞的说明了因为static成员会被共享,如果要调用两个不同的子类的时候,那个static成员的变量的值就会像一个全局变量一样不可控。他不同意。于是本着科学的精神,我们写下了一个简短的代码来验证:

  class A {

  protected static $var1 = null;

  public static function test(){

  echo get_called_class()。‘ ‘.static::$var1.‘<br/>‘;

  }

  }

  class B extends A {

  protected static $var1 = ‘b‘;

  }

  class C extends A {

  protected static $var1 = ‘c‘;

  }

  B::test();

  C::test();

  很显然,这次是我败了。 我期待的结果是c c,不过其实是b c。 那么这样看起来其实子类的static成员是只在子类这一层共享的。但是我总觉得不对劲,明明在写BaseModel的时候我已经又栽过跟头了,为什么这个验证出来并不支持我那个时候遇到的问题呢?于是我发现我记岔了。年轻多好。后来想起来,原来我这里不用static的原因仅仅是因为设计需要。

  我以为我错了。直到前几天又写了几个父子类(不是BaseModel了),大胆的用上了static成员,结果是轰轰烈烈的在自测中又摔了一跤。怎么回事!然后我仔细留意了一下自己这次的用法,将上面的例子改了一下运行:

  class A {

  protected static $var1 = null;

  protected static $var2 = null;

  public static function test(){

  if(!static::$var2){

  static::$var2 = static::$var1;

  }

  echo get_called_class()。‘ ‘.static::$var1.‘<br/>‘;

  }

  }

  class B extends A {

  protected static $var1 = ‘b‘;

  }

  class C extends A {

  protected static $var1 = ‘c‘;

  }

  B::test();

  C::test();

  结果是

  B b

  C b

  如果说上次的结论是对了,那么这次又怎么解释?这里明明就是表示$var2是A,B,C共享的。$var1和$var2的差别这样看起来仅仅是有声明和没声明的区别。于是我又改成这样:

  class A {

  protected static $var1 = null;

  protected static $var2 = null;

  public static function test(){

  if(!static::$var2){

  static::$var2 = static::$var1;

  }

  echo get_called_class()。‘ ‘.static::$var1.‘<br/>‘;

  }

  }

  class B extends A {

  protected static $var1 = ‘b‘;

  protected static $var2 = null;

  }

  class C extends A {

  protected static $var1 = ‘c‘;

  protected static $var2 = null;

  }

  B::test();

  C::test();

  结果是

  B b

  C c

  我当时内心是崩溃的。于是我上了Stack Overflow,发现栽坑的不止我一个。

  只有显式的声明出来的static成员才会被视为是只从属于子类的。

  只有显式的声明出来的static成员才会被视为是只从属于子类的。

  只有显式的声明出来的static成员才会被视为是只从属于子类的。

  重要的事情说三遍!不过如果子类很多的话, 动态决定值的成员 每个都这样去声明,就从写代码这件事上失去了用static的意义。一个更好的方法是,把$var2变成一个数组,每个类要用的值放在$var[__CLASS__]里面使用。

  不过不管怎么说,如非必要,还是尽量不用static成员继承吧。

  还有一个有点类似的“坑”。我们说到private成员的时候,都知道private是指私有的,不会被子类继承。但是有时候写代码的时候会忘记,直到载跟头了才想起来原来是private导致子类找不到该有的成员,或者说是private都在子类声明了,但是因为调用函数时是调用父类函数,结果得到的是父类这个private的值而不是子类的。遇到这种情况不可能又将函数原样的重写在子类里。所以使用private要特别小心。

  曾经在使用Rackspace的SDK的时候就看到有些类里面使用了private成员,但是由于他们给出了不必要的打开文件权限,导致代码在我们的服务器上运行不了。那么这个时候本想写个子类覆盖一下这个成员的初始值就好了,结果就因为这是个private成员,而最后需要把所有引用到的地方都拷到自己写的子类里面。为什么我们不直接改SDK,让成员变成protected?因为开发包也许下次就升级了呢?修正之后我们把子类移除就好了。如果修改库代码成了习惯,想升级的时候就没这么欢了。所以说,private成员的使用一定要慎之又慎,如果你也在开发SDK,就更需要考虑使用者是不是需要继承?如果你必须写private,你是不是能够保证代码能够适应各种场景的使用?

  除非你有非常充分的理由,static和private都是需要慎重使用的。

时间: 2024-10-28 23:34:10

谈谈PHP继承的那些事,我的个人见解的相关文章

谈谈一些有趣的CSS题目(四)-- 从倒影说起,谈谈 CSS 继承 inherit

开本系列,讨论一些有趣的 CSS 题目,抛开实用性而言,一些题目为了拓宽一下解决问题的思路,此外,涉及一些容易忽视的 CSS 细节. 解题不考虑兼容性,题目天马行空,想到什么说什么,如果解题中有你感觉到生僻的 CSS 属性,赶紧去补习一下吧. 不断更新,不断更新,不断更新,重要的事情说三遍. 所有题目汇总在我的 Github . 4.从倒影说起,谈谈 CSS 继承 inherit 给定一张有如下背景图的 div: 制作如下的倒影效果: 方法很多,但是我们当然要寻找最快最便捷的方法,至少得是无论图

【C#】OOP之继承那点事

前言: 继承这点事,说多不多,说少不少,这里只描述了一些我认为的基础篇,望各位大神指教.本节参照了C#高级编程和Think in java对继承的描述,我个人认为OOP只是思想,故看明白一个就说通的,只是语法上的区别. 权限限制关键字 类继承 类继承基本写法 类继承的好处 继承中的Virtual(虚方法) 抽象类 类继承的坏处 执行循序 不可继承的类 接口继承 1.类继承 类继承基本写法: public class SuperClass : Object{} public class SubCl

探讨 JS 的面向对象中继承的那些事

最近学了 JS 的面向对象,这篇文章主要是探讨 JS 的面向对象中继承的那些事. JS中继承的特点: 1.子类继承父类: 2.子类可以用父类的方法和属性 3.子类的改变可以不影响父类 下面用一个例子来说明 JS 的继承 这段代码创建了一个父类以及它的原型,同时还创建了一个子类,并继承了父类的私有属性 1 <script> 2 //这是父类 3 function Father(name,age,marry){ 4 this.name=name; 5 this.age=age; 6 this.ma

关于JavaScript继承的那些事

在JavaScript中,对象的创建可以脱离类型(class free),通过字面量的方式可以很方便的创建出自定义对象. 另外,JavaScript中拥有原型这个强大的概念,当对象进行属性查找的时候,如果对象本身内找不到对应的属性,就会去搜索原型链.所以,结合原型和原型链的这个特性,JavaScript就可以用来实现对象之间的继承了. 下面就介绍一下JavaScript中的一些常用的继承方式. 原型链继承 由于原型链搜索的这个特性,在JavaScript中可以很方便的通过原型链来实现对象之间的继

谈谈工作中的那些事,自动化、性能、接口测试!

今天可以说是一个特殊的日子,上午参加了集团测试小伙伴们的述职让自己存在颇多感触,还得弄个ppt几天后和他们一起分享.无奈~~~~说说那些事.        回想刚毕业的自己懵懵懂懂,领导问会不会性能.会不会自动化,我马上回答的肯定是:会!最后无论好差都交差了,不过回想起来再去看看那时候自己弄的东西时显得拿不出手.同感在今天的述职中也遇见了很多小伙伴都说到了这样的事情让我倍感亚历山大,不过心里在想还好几个大牛没来不然今天这个就坑了.遇见的基本如下:工作了近两年的A小伙伴上去说我这几个月做了什么收获

谈谈javascript继承

javascript实现继承大致有如下5种方式: 第1种,通过构造函数实现继承 function Parent() { this.name = 'parent' } function Child() { Parent.call(this) this.type = 'child' } 第2种,通过原型链实现继承 function Parent() { this.name = 'parent' } function Child() { this.type = 'child' } Child.prot

说一说js中__proto__和prototype以及原型继承的那些事

在面试中遇到过,问js如何实现继承,其实最好的方式就是构造函数+原型,今天在讨论中,发现自己以前理解上的一些误区,特地写出来,最近都比较忙,等手上的项目做完,可以来做个总结. 先说我以前没有认识到位的地方 1 __proto__这个不是标准实现的,但是各大浏览器和node都采用了,ECMA6计划标准化它,__proto__对应于标准中的[[prototype]],也就是所谓的内置原型属性,要把它和函数的prototype的相区别,其实,__proto__最终是指向Function.prototy

面试完 BAT等数十家公司,我想谈谈关于 Android 面试那些事

一. 本文目的 笔者将在本文中就Android开发工程师这一岗位,结合自己最近跳槽的经历,谈一谈自己对于面试的一些看法,希望能帮助到正在跳槽中的你们,也给自己的金三跳槽之旅划个圆满的分号. 注:本文适合工作三年以内的职场菜鸟观看,尤其针对想从小公司到大公司(更准确的说,是从普通公司到优质公司)的情况,大神或骨灰玩家可以关掉了. 二. 背景介绍 2.1 个人经历 抛开背景谈面试和offer是容易误导观众的,所以首先大概说说自己的情况供大家参考,某业内知名211本硕(不提母校名字,不给母校丢脸),计

java继承的那些事!

第二阶段 JAVA面向对象 第二章 继承 其实在我们面向对象这一阶段的学习中,我们就是围绕Java的三大特性(封装.继承.多态)来讲解的,在上一掌内容中我们讲解了一部分封装的有关知识,今天我们来重点学习继承的学习. 2.1 继承的基本认识 概述: 继承就是在一个已有类的基础上派生出新类 子类继承父类的特征和行为,使得子类对象(实例)具有父类的实例域和方法,或子类从父类继承方法,使得子类具有父类相同的行为 (父类更通用,子类更具体) #### 功能:有了继承以后,我们定义一个类的时候,可以在一个已