昨天我在写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. 【转载】Java实现word转pdf

    最近遇到一个项目需要把word转成pdf,GOOGLE了一下网上的方案有很多,比如虚拟打印.给word装扩展插件等,这些方案都依赖于ms word程序,在java代码中也得使用诸如jacob或jcom ...

  2. 『转』The Beginning of your Design Career

    想想,如果明天我开始学日语,坚持到毕业,其实也可以日语入门了.所以机会都是抓住,当初,也就是去年的时候,我那个时候就开始坚持日语入门,想想现在应该可以开始N2了吧-所以...过去不去理会,现在开始继续 ...

  3. 我的C++笔记(类与对象)

    /* * Main.cpp * * Created on: 2015-7-24 * Author: feiruo */ /* * 类与对象: * * 1.抽象: * 面向对象方法中的抽象,是指对具体问 ...

  4. RabbitMQ学习之spring-amqp的重要类的认识

    对于大多数应用来说都做了与spring整合,对于rabbitmq来说.也有与spring的整合.可能通过spring的官网找到spring-amqp项目下载.spring-amqp项目包括三个子项目: ...

  5. 347. 前K个高频元素

    题目描述 给定一个非空的整数数组,返回其中出现频率前 k 高的元素. 示例 1: 输入: nums = [1,1,1,2,2,3], k = 2 输出: [1,2] 示例 2: 输入: nums = ...

  6. H3C交换机telnet服务认证模式配置

    以H3C交换机为例,介绍telnet服务的三种认证方式配置(none无需认证,password密码认证,scheme账户+密码认证) None认证模式配置步骤:[H3C]telnet server e ...

  7. Linux常用命令速查

    索引表格 命令 功能简述 目录与文件基本操作 pwd 显示当前目录 ls 列出目录和文件名称 cp 复制文件或目录 mv 移动或更名现有的文件或目录 rm 删除文件或目录 mkdir 新建目录 rmd ...

  8. SQL SEVER (ROLLUP与CUBE,ROW_NUMBER())使用方法

    1.建立测试专用数据: if object_id('TESTDB') is not null drop table TESTDB ), B INT) insert into TESTDB union ...

  9. 安装ubuntu14.04之后要做的一些事

    前言: 用ubuntu14.04也有一段时间了,感觉与之前版本相比还是在挺多方面有了改进.但刚装完还是有一些小问题需要自己动手解决.鉴于网上的内容太过零碎,有些方案也太过老旧,因此在这里为大家总结一些 ...

  10. VUE 利用 webpack 给生产环境和发布环境配置不同的接口地址

    转载地址: https://blog.csdn.net/gebitan505/article/details/58166055 VUE 利用 webpack 给生产环境和发布环境配置不同的接口地址 前 ...