昨天我在写PHP程序的时候,无意发现在PHP的::操作符非常强大,不仅仅是只用在访问parent,sel,静态成员属性、常量上面,其实他的功能强大了去了

这个符号在PHP中实际上叫做范围解析符,这是PHP官方的解释php.net/manual/zh/language.oop5.paamayim-nekudotayim.php。但是官方中文部分只是简单的介绍了

访问parent,sel,静态成员属性、常量的作用,而下面英文部分,可能由于咱们这些吊死对英文不好的缘故,就没看下去了,实现上,下面还有一个强大的特性,我们大家都没看到。
什么叫范围操作符呢,可能很多人都没有认真的去分析过,这几个字,根据我这两年的编程的理解,我觉得应该是定义了类中的成员属性和成员方法的所在范围,感觉有点类似命名空间的感觉。

而self::method();其实指的就是调用当前范围里的method方法,那parent自然指的就是父类所在范围,这就是为什么,我们在子类中
覆盖父类的方法,我们仍然可以用parent::method()访问的原因。大家还要注意一点,self永远指向的是其实声明位置所在类本身和而
parent永远指向的是声明位置所在类的父类,不会受继承上下文的影响而改变。
前面讲的这些可能大家有些人已经知道了,上面说过::在PHP中叫范围解析操作符,上面我说过了既然,parent和self都能指向一个类范围的代名词,那试想下,我们能不能用类名指向某个范围呢,答案当然是可以的。我们可以这样

  1. class a{
  2. public function demo(){
  3. echo 'abc';
  4. }
  5. }
  6. a::demo();
不知道大家发现没,里面的demo方法并不静态方法,但我却可以这样访问,那是因为a::指向的class范围,所以可以访问时面的demo方法,PHP官方有这样的示例代码,就在上面我发的链接中,大家可以仔细看下。

大家肯定又疑惑这跟多继承有什么关系呢,别急,心急吃不了热豆腐的!!!!!

其实这个范围操作符还有一个很大的特性,就是当在类外部调用一个非静态公共方法时,方法内部的$this指针会指向他当前运行所在对象本身(宿主对象),
当一个类实例化时,方法里面$this永远指向的实例化后的对象本身,为什么呢,因为这个方法是被这个对象调用执行的,所以这个对象就是他的宿主对象。而
刚才我们直接在全局调中调用这个方法时,由于这个方法运行时不在任何对象中,所以不存在任何宿主对象,所以如果在方法里面写$this的话会报致命错误。
那怎么样才能拥有宿主对象,然后直接调用。仔细看我刚才讲的同学可能注意到,我刚才说过"因为这个方法是被这个对象调用执行的,所以这个对象就是他的宿主
对象"。所以理论上说,我们只要改变执行这个方法的所在环境,就可以改变他的宿主对象。下面我们来实验下

class a{
        public function say_name(){
                echo 'my name is'.$this->name;
        }
}

class b{
        public $name='xbs530';
        public function run(){
                a::say_name();
        }
}

$o=new b;
$o->run();
大家猜会输出什么。。。。。。。
大家可以动手做一下试验。。。。。

由于a::say_name()是被对象$o调用的,所以方法里面的$this自然指向了对象$o。很神奇吧。。。。

这个我之前发的那个PHP范围解析操作符中都有讲,好了基础就讲到这里,下面直接上多继承的代码了,欢迎吐槽。。。。。

/*
        多继承基类
                注:要想实现类多继承,必需首先继承此基类
*/
        
class multi_extends{
        
        public function __construct(){
                $this->_init_extends();
        }
        
        protected function _init_extends(){
                if(property_exists($this,'_extends'))
                {
                        $extends=&$this->_extends;
                        foreach($extends as $class)
                        {
                                $this->_extends($class);
                        }
                }
        }
        
        public function _extends($class_name){
                //分析目标
                $ref = new ReflectionClass($class_name);
                
                //继承公共属性
                $property_list=$ref->getProperties(ReflectionProperty::IS_PUBLIC);
                foreach($property_list as $property)
                {
                        $property_name=$property->name;
                        $property_value=$property->getValue(new $class_name);
                        if($property_name==='_extends')
                        {
                                foreach($property_value as $c)
                                {
                                        $this->_extends[]=$c;
                                }
                        }else{
                                if(!property_exists($this,$property_name))
                                {
                                        $this->{$property->name}=$property_value;
                                }
                        }
                }
                
                //继承公共方法
                $method_list=$ref->getMethods(ReflectionMethod::IS_PUBLIC);
                foreach($method_list as $method)
                {
                        $this->_extends_method[$method->name]=$method->class;
                }
        }
        
        function __call($m,$a){
                if($c=$this->_extends_method[$m])
                {
                        eval("$c::$m(".'$a'.");");
                }
        }
}

header('content-type:text/html;charset=utf-8');

class a{
        public $a=5;
        
        public function say_name(){
                echo '我叫'.$this->name.'<br>';
        }
}

class b{
        function say_age(){
                echo '我今年'.$this->age.'岁<br>';
        }
        
}

class c extends multi_extends{
        public $_extends=array('a','b');
        
        
        public function say_hellow(){
                echo 'hellow '.$this->name.'<br>';
        }
}

class d{
        function say_goodbye(){
                echo 'goodbye '.$this->name.'<br>';
        }
        
}

class persion extends multi_extends{
        public $_extends=array('c','d');
        
        
        public $name="熊宝山";
        public $age="20";
        
        
        
}

//执行继承后的方法
$o=new persion();
$o->say_name();
$o->say_age();
$o->say_hellow();
$o->say_goodbye();

php利用反射真正实现多继承(非接口模拟)的更多相关文章

  1. Atitit利用反射获取子类 集合 以及继承树

    Atitit利用反射获取子类 集合 以及继承树 想从父类往下找子类的确是不可能的,要知道只要类不是final的话谁都有继承它的自由不需要事前通知父类. Eclipse实现不是重父类开始找而是重子类往回 ...

  2. Python基础篇【第3篇】: Python异常处理、反射、动态导入、利用反射的web框架

    异常处理 什么是异常? 异常即是一个事件,该事件会在程序执行过程中发生,影响了程序的正常执行. 一般情况下,在Python无法正常处理程序时就会发生一个异常.异常是Python对象,表示一个错误.当P ...

  3. c#反射机制学习和利用反射获取类型信息

    反射(Reflection)是.NET中的重要机制,通过放射,可以在运行时获得.NET中每一个类型(包括类.结构.委托.接口和枚举等)的成员,包括方法.属性.事件,以及构造函数等.还可以获得每个成员的 ...

  4. 静态代理和利用反射形成的动态代理(JDK动态代理)

    代理模式 代理模式的定义:为其他对象提供一种代理以控制对这个对象的访问.在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用. 静态代理 1.新建 ...

  5. [.net 面向对象程序设计进阶] (20) 反射(Reflection)(上)利用反射技术实现动态编程

    [.net 面向对象程序设计进阶] (20) 反射(Reflection)(上)利用反射技术实现动态编程 本节导读:本节主要介绍什么是.NET反射特性,.NET反射能为我们做些什么,最后介绍几种常用的 ...

  6. <五>JDBC_利用反射及JDBC元数据编写通用的查询方法

    此类针对javaBean类写了一个通用的查询方法,List<javaBean> 通用查询更新中...:通过学习,深刻体会到学会反射就等于掌握了java基础的半壁江山! 一.使用JDBC驱动 ...

  7. .NET 利用反射将对象数据添加到数据库

    .NET 利用反射将对象数据添加到数据库   一些小型的项目,在不使用其他的框架(LINQ,NHibernate,EF等等框架)的前提下,这时候一些反复的增删改查就会让我们感到极其的繁琐,厌烦,为了避 ...

  8. 第十篇 一个利用反射实现的Excel导出

    前些天写了个导出Excel的公共类,因为项目中也需要上传Excel,没有利用Office组件,(PS:Office在上传文件时候,Excel进程无法关闭,会导致上传和打开失败)有的说利用Kill把进程 ...

  9. 优化Servlet:(利用反射的思想)

    1.创建BaseServlet (重写父类的service方法) package com.learning.web.servlet; import java.io.IOException; impor ...

随机推荐

  1. 图片文字滚动插件jQuery Scrollbox

    图片文字滚动插件jQuery Scrollbox附件中提供了五种图片.文字滚动样式,只需调用jquery库和jQuery Scrollbox插件,然后再加一段简单的jquery代码即可使用,兼容性良好 ...

  2. 时间&物质&效率

    由于我的家庭是地道的农民家庭,在上学的时候,父母很辛苦的供我读初中,高中,大学. 现在我想说的是,用时间来换取效率是我求学时最大的遗憾. 举一个例子吧:每次回家坐火车,火车很费时间,假如我不缺钱,完全 ...

  3. 几个概念:x86、x86-64和IA-32、IA-64

    最近在学习操作系统方面的知识,学习操作系统难免要和CPU打交道,虽然现在CPU和操作系统不像计算机发展初期一样是绑定在一起的,但是大家都知道操作系统和CPU Architecture的联系是很紧密的, ...

  4. Vtk读取并显示保存图像

    (1):Vtk读取并显示图像:三种方法 转载:未知出处 转载:用VTK显示平面图片:http://blog.csdn.net/tonylk/article/details/464881 用到vtkJP ...

  5. ES6的let命令

    1.let命令所在的代码块内有效: 2.循环的计数器,就很合适使用let命令:(for循环还有一个特别之处,就是循环语句部分是一个父作用域,而循环体内部是一个单独的子作用域.) 例如:如下会输出十次s ...

  6. 对服务器磁盘、CPU、内存使用状态,设置163邮件告警

    1,桥接模式可上网,首先你的邮箱已经开通yum -y install mailx dos2unix.x86_64  mailx -V[root@localhost ~]# vim /etc/mail. ...

  7. win10更新后程序路径盘符变成*星号解决方法

    发现这个问题是当我在命令行里输入java -version时提示 找不到*:\Program Files\Java...之类的 怎么好好的D:\Program Files\Java变成了*:\Prog ...

  8. 复习MySQL①创建数据库及数据表

    • 创建数据库:create database 数据库名称; – 例:创建名为test的测试数据库 create database test; • 查看创建好的数据库:show create data ...

  9. GDI Bezier 样条曲线(7)

    Bezier 样条曲线 Bezier 样条曲线使用四个点来定义:两个端点(起点和终点)和两个控点(用于使其不同程度地弯曲). 绘制 Bezier 样条曲线 使用 PolyBezier 函数和 Poly ...

  10. .net 技术基础

    C#常见运算符 一元运算符(+.-.!.~.++.--) 算术运算符(*./.%.+ . – ) 移位运算符(<< .>> ) 关系和类型测试运算符(==.!=.<.&g ...