zl程序教程

您现在的位置是:首页 >  后端

当前栏目

php面向对象的方法重载两种版本比较

PHP方法 版本 比较 两种 面向对象 重载
2023-06-13 09:14:02 时间
多个函数用同一个名字,但参数表,即参数的个数或(和)数据类型可以不同,调用的时候,虽然方法名字相同,但根据参数表可以自动调用对应的函数。

PHP4中仅仅实现了面向对象的部分的、简单的功能,而PHP5以后对对象的支持就强大的多了。

对于多态的实现,PHP4只支持覆盖(override),而不支持重载(overload)。但我们可以通过一些技巧来“模拟”重载的实现。

PHP5虽然可以支持覆盖和重载,但重载在具体实现上,和其他语言还有较大的差别。

1,在PHP4中“模拟”重载

试看以下代码:

<?php
//根据参数个数选择执行不同的方法(在PHP4中模拟"重载"(多态的一种)

classMyclass
{
functionMyclass()
{
$method="method".func_num_args();
$this->$method();
}

functionmethod1($x)
{
echo"method1";
}

functionmethod2($x,$y)
{
echo"method2";
}
}

//通过在类中的额外的处理,使用这个类对用户是透明的:
$obj1=newMyclass("A");//将调用method1
$obj2=newMyclass("B","C");//将调用method2
?>

以上代码中,通过在构造函数中使用func_num_args()函数取到参数的个数,自动执行method1或method2方法。我们可以结合函数func_get_arg(i)和func_get_args()对以上示例进行改进。

2,在PHP5中使用重载

先看以下示例:

复制代码代码如下:

<?php
classMyclass
{
public$attriable;
public$one="thisisone";
public$two="thisistwo";

function__construct()
{
}

functionone($one)
{
$this->one=$one;
$this->attriable=$this->one;
}

functionone($one,$two)
{
$this->one=$one;
$this->two=$two;
$this->attriable=$this->one.$this->two;
}

functiondisplay()
{
echo$this->attriable;
}
}

$one="thisismyclass";
$two="Imthebest";
$myclass=newmyclass();
$myclass->one($one);
$myclass->display();

$myclass->one($one,$two);
$myclass->display();

//本例的做法,在PHP中是不正确的!
?>


使用过C++、Java、C#重载的人,很习惯地就会写出以上的重载实现的PHP代码。但这在PHP5中是不正确的。PHP5并不是对前述几种语言的模仿,而是有自己的一套实现方法重载的方法(是好是坏,这里不讨论)。虽说PHP5的类较PHP4强大了许多,但是在“重载”这个问题上并没有像我们预期的那样“改善”。在“强”类型的语言中可以通过不同的参数类型来实现“重载”,比如C++、Java、C#等。在“固定参数”传递的语言中,还可以通过参数的个数进行传递,比如Java,但是PHP是弱类型语言,因此不会出现类似以上的“重载”。

PHP5中重载可以通过__get,__set,and__call几个特殊方法来进行。当Zend引擎试图访问一个成员并没有找到时,PHP将会调用这些方法。

在以下示例中,__get和__set代替所有对属性变量数组的访问。如果必要,你还可以实现任何类型你想要的过滤。例如,脚本可以禁止设置属性值,在开始时用一定的前缀或包含一定类型的值。__call方法说明了你如何调用未经定义的方法。你调用未定义方法时,方法名和方法接收的参数将会传给__call方法,PHP传递__call的值返回给未定义的方法。

复制代码代码如下:

<?php
classOverloader
{
private$properties=array();

function__get($property_name)
{
if(isset($this->properties[$property_name]))
{
return($this->properties[$property_name]);
}
else
{
return(NULL);
}
}

function__set($property_name,$value)
{
$this->properties[$property_name]=$value;
}

publicfunction__call($method,$p)
{
print("Invoking$method()<br>\n");
//print("Arguments:");
//print_r($args);
if($method=="display")
{
if(is_object($p[0]))
$this->displayObject($p[0]);
else
if(is_array($p[0]))
$this->displayArray($p[0]);
else
$this->displayScalar($p[0]);
}
}

publicfunctiondisplayObject($p)
{
echo("你传入的是个对象,内容如下:<br>");
print_r($p);
echo"<hr>";
}

publicfunctiondisplayArray($p)
{
echo("你传入的是个数组,内容如下:<br>");
print_r($p);
echo"<hr>";
}

publicfunctiondisplayScalar($p)
{
echo("你传入的是个单独变量,内容如下:<br>".$p);
echo"<hr>";
}
}

$o=newOverloader();

//调用__set()给一个不存在的属性变量赋值
$o->dynaProp="DynamicContent";

//调用__get()
print($o->dynaProp."<br>\n");

//调用__call()
//$o->dynaMethod("Leon","Zeev");

$o->display(array(1,2,3));
$o->display("Cat");
?>


以上代码中,调用了display()方法,可以根据参数的类型和个数调用类中的对应的代码段,从而实现了对象方法的重载。