<?php
class thiswillfail {
public function foo() {
return "hello world!";
}
}
$class = new thiswillfail;
$class->bar();
?>
<?php
//装入pear的 <a href="http://pear.php.net/package/db/">db package</a>
require_once "db.php";
class persistable {
private $data = array();
private $table = "users";
public function __construct($user) {
$this->dbh = db::connect("mysql://user:[email protected]/database");
$query = "select id, name, email, country from " .
$this->table . " where name = ?";
$this->data = $this->dbh->getrow($query, array($user),
db_fetchmode_assoc);
}
public function __get($member) {
if (isset($this->data[$member])) {
return $this->data[$member];
}
}
public function __set($member, $value) {
// dataset的id是只读的
if ($member == "id") {
return;
}
if (isset($this->data[$member])) {
$this->data[$member] = $value;
}
}
public function __destruct() {
$query = "update " . $this->table . " set name = ?,
email = ?, country = ? where id = ?";
$this->dbh->query($query, $this->name, $this->email,
$this->country, $this->id);
}
}
$class = new persistable("martin jansen");
$class->name = "john doe";
$class->country = "united states";
$class->email = "[email protected]";
?>
你遇到的第一个问题可能是__construct(),这是php 5中引入的新的构造器方法。在php 4时代,构造器总是与它们的类名相匹配。在php 5中已不再是这样。你不需要对构造器方法有过多的了解,除了调用它可以创建一个类的实例外;并注意到,这里使用了一个参数 - 执行一个基于此参数的数据库。此构造器把查询结果赋值给类属性$data。
接下来,程序定义了两个特别的方法__get()和__set()。你应该对它们早已熟悉:__get()用于读取未定义的属性值,__set()用于修改未定义的属性值。
这意味着无论什么时候从持续性存储类中读取/写入一个未定义的属性,由这些专门方法来负责管理在属性数组变量$data中的信息,而不是直接改变类的属性(切记:变量$data包含着来自于数据库中的一行!)。
类中的最后一个方法是__construct()的对立者- 析构器__destruct()。php在"脚本关闭阶段"调用析构器,典型地这是在php脚本执行快要结束的时候。析构器把来自于$data属性的信息写回到数据库中去。这正是前面同步(synchronization )术语的含义。
你可能早已注意到,这里的代码使用了pear的数据库抽象层包(database abstraction layer package)。其实这无所谓,通过别的方式与数据库通讯也一样能说明本文的主题。
如果你细心观察,会发现该持续性存储类的描述比较简单。例子中仅涉及了一个数据库表,而没有考虑更复杂的数据模型,如使用left join和其它复杂的数据库操作技术。然而你不必受此约束,借助于属性重载,你可以使用你自己理想的数据库模型。只需要加入少许代码,你即可以在该持续性存储类中运用复杂的数据库特性。
还存在一个小问题 - 当在析构器中查询失败时并没有引入错误处理机制。是析构器的天性导致在这种情况下不可能显示相应的错误信息,因为构建html标志常常在php调用构析器之前就已经结束了。
为解决这个问题,你可以把__destruct()重命名为象savedata()这样的名字并在调用脚本的某处手工执行这一方法。这对于类的持续性存储的概念并没有任何改变;仅是多写几行代码而已。作为选择,你还可以在析构器中使用函数error_log()来记录下属于系统范围的错误记录文件中的错误信息。
属性重载的工作机制就是这样。下面我们讨论一下方法重载。
四、方法重载举例
1. 动态的getter/setter方法
下列代码实现了"动态"getter/setter方法以借助于方法重载的帮助来控制类。下面我们结合源代码进行分析:
<?php
class dynamicgettersetter {
private $name = "martin jansen";
private $starbucksdrink = "caramel cappuccino swirl";
function __call($method, $arguments) {
$prefix = strtolower(substr($method, 0, 3));
$property = strtolower(substr($method, 3));
if (empty($prefix) || empty($property)) {
return;
}
if ($prefix == "get" && isset($this->$property)) {
return $this->$property;
}
if ($prefix == "set") {
$this->$property = $arguments[0];
}
}
}
$class = new dynamicgettersetter;
echo "name: " . $class->getname() . "/n";
echo "favourite starbucks flavour: " . $class->getstarbucksdrink() . "/n/n";
$class->setname("john doe");
$class->setstarbucksdrink("classic coffee");
echo "name: " . $class->getname() . "/n";
echo "favourite starbucks flavour: " . $class->getstarbucksdrink() . "/n/n";
?>
$class->thismethoddoesnotexist("martin", 42);
/导向__call()的第二个参数
array
(
[0] => martin
[1] => 42
)
新闻热点
疑难解答