PHP的面向对象编程
发布人:shili8
发布时间:2022-12-01 01:17
阅读次数:26
面向对象编程的概念:
不同的作者之间说法可能不一样,但是一个oop语言必须有以下几方面:
抽象数据类型和信息封装?
继承?
多态?
在php中是通过类来完成封装的:?
<?php?
class?something?{?
//?在oop类中,通常第一个字符为大写?
var?$x;?
function?setx($v)?{?
//?方法开始为小写单词,然后使用大写字母来分隔单词,例如getvalueofarea()?
$this->x=$v;?
}?
function?getx()?{?
return?$this->x;?
}?
}?
当然你可以按自已的喜好进行定义,但最好保持一种标准,这样会更有效。数据成员在类中使用"var"声明来定义,在给数据成员赋值之前,它们是没有类型的。一个数据成员可以是一个整数,一个数组,一个相关数组(associative?array)或者是一个对象。方法在类中被定义成函数形式,在方法中访问类成员变量时,你应该使用$this->name,否则对一个方法来说,它只能是局部变量。?
使用new操作符来创建一个对象:?
$obj=new?something;?
然后你可以使用成员函数通过:?
$obj->setx(5);?
$see=$obj->getx();?
在这个例子中,setx成员函数将5赋值给对象的成员变量x(不是类的),然后getx返回它的值5。可以象:$obj->x=6那样通过类引用方式来存取数据成员,这不是一个很好的oop习惯。我强烈建议通过方法来存取成员变量。如果你把成员变量看成是不可处理的,并且只通过对象句柄来使用方法,你将是一个好的oop程序员。不幸的是,php不支持声明私有成员变量,所以不良代码在php中也是允许的。继承在php中很容易实现,只要使用extend关键字。?
<?php?
class?another?extends?something?{?
var?$y;?
function?sety($v)?{?
$this->y=$v;?
}?
function?gety()?{?
return?$this->y;?
}?
}?
"another"类的对象现在拥有了父类(something)的全部的数据成员及方法,而且还加上了自已的数据成员和方法。
你可以使用
$obj2=new?something;
$obj2->setx(6);
$obj2->sety(7);?
php现在还不支持多重继承,所以你不能从两个或两个以上类派生出新的类来。你可以在派生类中重定义一个方法,如果我们在"another"类中重定义了getx方法,我们就不能使?用"something"中的getx方法了。如果你在派生类中声明了一个与基派同名的数据成员,那么当你处理它时,?它将“隐藏”基类的数据成员。
你可以在你的类中定义构造函数。构造函数是一个与类名同名的方法,当你创建一个类的对象时会被调用,例如:?
<?php?
class?something?{?
var?$x;?
function?something($y)?{?
$this->x=$y;?
}?
function?setx($v)?{?
$this->x=$v;?
}?
function?getx()?{?
return?$this->x;?
}?
}?
所以你可以创建一个对象,通过:?
$obj=new?something(6);?
构造函数会自动地把6赋值给数据变量x。构造函数和方法都是普通的php函数,所以你可以使用缺省参数。?
function?something($x="3",$y="5")?
接着:?
$obj=new?something();?//?x=3?and?y=5?
$obj=new?something(8);?//?x=8?and?y=5?
$obj=new?something(8,9);?//?x=8?and?y=9?
缺省参数使用c++的方式,所以你不能忽略y的值,而给x一个缺省参数,参数是从左到右赋值的,如果传入的参数少于要求的参数时,其作的将使用缺省参数。?
当一个派生类的对象被创建时,只有它的构造函数被调用,父类的构造函数没被调用,如果你想调用基类的构造函数,你必须要在派生类的构造函数中显示调用。可以这样做是因为在派生类中所有父类的方法都是可用的。?
<?php?
function?another()?{?
$this->y=5;?
$this->something();?
//显示调用基类构造函数?
}?
oop的一个很好的机制是使用抽象类。抽象类是不能实例化,只能提供给派生类一个接口。设计者通常使用抽象类来强迫程序员从基类派生,这样可以确保新的类包含一些期待的功能。在php中没有标准的方法,但是:如果你需要这个特性,可以通过定义基类,并在它的构造函数后加上"die"?的调用,这样就可以保证基类是不可实例化的,现在在每一个方法(接口)后面加上"die"?语句,所以,如果一个程序员在派生类中没有覆盖方法,将引发一个错误。而且因为php?是无类型的,你可能需要确认一个对象是来自于你的基类的派生类,那么在基类中增加一个方法来实义类的身份(返回某种标识id),并且在你接收到一个对象参数时校验这个值。当然,如果一个邪恶不好的程序员在派生类中覆盖了这个方法,这种方法就不起作用了,不过一般问题多发现在懒惰的程序员身上,而不是邪恶的程序员。
当然,能够让基类对程序员无法看到是很好的,只要将接口打印出来做他们的工作就可以了。在php中没有析构函数。
重载(与覆盖不同)在php中不支持。在oop中,你可以重载一个方法来实现两个或重多的方法具有相同的名字,但是有不同数量或类型的参数(这要看语言)。php?是一种松散类型的语言,所以通过类型重载不起作用,然而通过参数的个数不同来重载也不起作用。?
有时在oop中重载构造函数非常好,这样你可以通过不同的方法创建对象(传递不同数量的参数)。在php中实现它的技巧是:?
<?php?
class?myclass?{?
function?myclass()?{?
$name="myclass".func_num_args();?
$this->$name();?
//注意$this->name()一般是错误的,但是在这里$name是一个将被调用方法的名字?
}?
function?myclass1($x)?{?
code;?
}?
function?myclass2($x,$y)?{?
code;?
}?
}?
通过在类中的额外的处理,使用这个类对用户是透明的:?
$obj1=new?myclass('1');?//将调用myclass1?
$obj2=new?myclass('1','2');?//将调用myclass2?
有时这个非常好用。
多态?
多态是对象的一种能力,它可以在运行时刻根据传递的对象参数,决定调用哪一个对象的方法。例如,如果你有一个figure的类,它定义了一个draw的方法。并且派生了circle和rectangle?类,在派生类中你覆盖了draw方法,你可能还有一个函数,它希望使用一个参数x,并且可以调用$x->draw()?。如果你有多态性,调用哪个draw方法就依赖于你传递给这个函数的对象类型。?