欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

PHP的面向对象编程

程序员文章站 2022-07-21 21:10:02
面向对象编程的概念: 不同的作者之间说法可能不一样,但是一个oop语言必须有以下几方面: 抽象数据类型和信息...

面向对象编程的概念:

不同的作者之间说法可能不一样,但是一个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方法就依赖于你传递给这个函数的对象类型。