php利用反射真正实现多继承(非接口模拟)

本文详细解析PHP中的范围解析操作符(::),包括其基本用法、作用范围、多继承应用及如何通过该操作符访问类方法。重点展示了如何利用::操作符在不同范围内调用方法,并通过实例演示了多继承场景下的方法调用机制。通过实验代码,深入浅出地阐述了方法执行时的$this指向及其如何改变宿主对象。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

昨天我在写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都能指向一个类范围的代名词,那试想下,我们能不能用类名指向某个范围呢,答案当然是可以的。我们可以这样

<?php
class a {
	public function demo() {
		echo 'abc';
	}
}
a::demo();


不知道大家发现没,里面的demo方法并不静态方法,但我却可以这样访问,那是因为a::指向的class范围,所以可以访问时面的demo方法,PHP官方有这样的示例代码,就在上面我发的链接中,大家可以仔细看下。
大家肯定又疑惑这跟多继承有什么关系呢,别急,心急吃不了热豆腐的!!!!!
其实这个范围操作符还有一个很大的特性,就是当在类外部调用一个非静态公共方法时,方法内部的$this指针会指向他当前运行所在对象本身(宿主对象),当一个类实例化时,方法里面$this永远指向的实例化后的对象本身,为什么呢,因为这个方法是被这个对象调用执行的,所以这个对象就是他的宿主对象。而刚才我们直接在全局调中调用这个方法时,由于这个方法运行时不在任何对象中,所以不存在任何宿主对象,所以如果在方法里面写$this的话会报致命错误。那怎么样才能拥有宿主对象,然后直接调用。仔细看我刚才讲的同学可能注意到,我刚才说过"因为这个方法是被这个对象调用执行的,所以这个对象就是他的宿主对象"。所以理论上说,我们只要改变执行这个方法的所在环境,就可以改变他的宿主对象。下面我们来实验下
<?php
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范围解析操作符中都有讲,好了基础就讲到这里,下面直接上多继承的代码了,欢迎吐槽。。。。。
<?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();


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值