在回顾面向对象的基本概念之后,我将介绍一些更高级的技术 。
串行化
PHP并不支持持久性对象,在面向对象语言中持久性对象是一些经过应用程序 多次调用仍然保持其状态和功能 的对象,这意味着有一种能保存对象到文件 或数据库 中然后重新装载对象。这种机制称之为串行化。PHP 有一个串行化函数 ,可以在对象中调用,串行化函数返回一个字符 串代表这个对象。然后串行化函数保存的是成员数据 而不是成员函数。
在PHP4中,如果你串行化一个对象到字符串 $s, 然后删除此对象,再反串行化对象到 $obj, 你仍然可以调用对象的方法函数。但我不推荐这种方法,这因为(a)这种功能在将来不一定支持(b)这导致一种幻象,如果你保存串行化对象到磁盘并退出程 序。将来重新运行此脚本时你不能反串行化此对象并希望对象的方法函数仍有效,因为串行化出来的字符串并没有表示任何成员函数。最后,串行化保存对象的成员 变量在PHP中非常有用,仅仅如此. (你可以串行化联合数组和数组到磁盘里)。
例子 :
$obj=new Class foo();
$str=serialize($obj);
// Save $str to disk
//...some months later
//Load str from disk
$obj2=unserialize($str)
?>
上例中,你可以恢复成员变量而没有成员函数(根据文档)。这导致 $obj2->x 是唯一的方法来存取成员变量(因为没有成员函数)。
这里还有一些方法解决这个问题,但我留下给你因为它会搞脏这个干净的文档。
我希望PHP将来能全面支持串行化。
使用类来操纵保存的数据
PHP和面向对象编程中一个比较好的地方是你很容易定义类来操纵某些东西,并且当需要时调用合适的类。假设有一个HTML文件,你需要通过选择产品的 ID号来选择一个产品,你的数据保存在数据库中,而你想显示产品的信息,如价格等等。你有不同种类的产品,同样的动作对不同的产品有不同的含义。
例如,显示一个声音意味着播放它,而对其他产品来说可能是显示一个存储在数据库的图片。你可以用面向对象编程和PHP来达到,代码 少但更好。
定义一个类,定义类应该有的方法,然后通过继承来定义每一种产品的类(SoundItem类, ViewableItem类,等等),重定义每个产品类的方法,使它们如你所需。根据你保存在数据库中的表的产品类型字段来给每一种产品类型定义一个类, 一个典型的产品表应有字段(id, 类型, 价格, 描述,等等)。
在脚本中你从数据库的表中获取 类型信息,然后实例化相应类的对象:
$obj=new $type();
$obj->action();
?>
这是PHP比较的特性,你可以调用 $obj 的显示方法或其他方法而不用去管对象的类型。通过这种技术,当你增加一种新类型的对象时,你不需要去修改脚本。这个方法有点威力,就是定义所有对象应有的 方法而不管它的类型,然后在不同的类中以不同的方式来实现,这样就可以在脚本中对不同的类型对象使用他们,再没有 if, 没有两个程序员在同一个文件里,永远快乐。你相信编程是这样快乐不?维护代价小并且可重用?
如果你带领一组程序员,最好的方法是划分任务,每人可以对某种类和对象负责。国际化可以用同样的技术解决,使合适的类对应使用者选择的不同的语言等等。
复制和克隆
当你创建一个对象 $obj, 你可以使用 $obj2 = $obj 来拷贝一个对象,新的对象是 $obj 的一个拷贝(不是引用),所以在赋值完新对象有 $obj 同新的状态。有时候你不想这样,只想创建和 obj 同样的新对象,调用新对象的构造函数如同你曾使用过 new 命令 。这可以通过PHP的串行化和使用基类并且其他类必须从基类继承来达到。
进行危险的地带
当你串行化一个对象,你得到一个有特定格式的字符串,如果你有好奇心,可能你会探寻其中的秘密,字符串中有一个东西就是类的名字,你可以解开它:
$herring=serialize($obj);
$vec=explode(:,$herring);
$nam=str_replace("/"",/,$vec[2]);
?>
假设你创建一个类 "Universe" 并且使所有类都从 "Universe" 继承而来,你可以在 "Universe" 定义一个克隆的方法:
class Universe {
function clone() {
$herring=serialize($this);
$vec=explode(:,$herring);
$nam=str_replace("/"",/,$vec[2]);
$ret=new $nam;
return $ret;
}
}
//Then:
$obj=new Something();
//Something extends Universe !!
$other=$obj->clone();
?>
你所得的是类 Something 的新对象如同使用 new 一样,并且构造函数被调用等等。我不知道这对你是不是有用,这是一个很好的实践,Universe 类知道它的继承类的名字。对你来说,唯一的限制是你的想象力!!!
注意:我使用的是PHP4, 文章里有些东西可能不适合PHP3。