当前位置: 首页 > 图文教程 > 网络编程 > PHP > PHP 5.0 中的对象重载技术研究

PHP
PHP 手机归属地查询 api
php 自写函数代码 获取关键字 去超链接
检查url链接是否已经有参数的php代码 添加 ? 或 &
PHP生成网页快照 不用COM不用扩展.
一步一步学习PHP(1) php开发环境配置
一步一步学习PHP(2):PHP类型
一步一步学习PHP(3) php 函数
一步一步学习PHP(4) php 函数 补充2
提高PHP编程效率 引入缓存机制提升性能
php 数组的合并、拆分、区别取值函数集
PHP采集相关教程之一 CURL函数库
IP138 IP地址查询小偷实现代码
php 生成静态页面的办法与实现代码详细版
一步一步学习PHP(5) 类和对象
一步一步学习PHP(6) 面向对象
Apache环境下PHP利用HTTP缓存协议原理解析及应用分析
PHP 截取字符串函数整理(支持gb2312和utf-8)
php foreach 使用&(与运算符)引用赋值要注意的问题
PHP IPV6正则表达式验证代码
用PHP ob_start()控制浏览器cache、生成html实现代码

PHP 5.0 中的对象重载技术研究


出处:互联网   整理: 软晨网(RuanChen.com)   发布: 2009-11-03   浏览: 24 ::
收藏到网摘: n/a

 

  文/朱先忠编译

    一、简介

  很幸运,PHP 5.0中引入了对象重载技术。本文将探讨对于方法__call(),__set()以及__get()进行重载的可能性。在对重载理论作简单介绍后,我们将通过两个例子直奔主题:第一例,实现持续存储类;第二例,找到一种实现动态的getter/setter的方法。

  二、什么是对象重载?

  在PHP中谈到对象重载时,我们要区别两种类型:

  ·方法重载

  ·属性重载

  在方法重载的情况下,我们要定义一个魔术般的方法__call(),它将实现一个在相应类中对未定义方法的笼统调用。只有当你想存取类中未定义的方法时,这种笼统方法才会被调用。在没有方法重载的情况下,下面的例子将导致PHP显示一条致命错误信息:Call to undefined method ThisWillFail::bar() in/some/directory/example.php on line 9并流产程序的执行:

<?php
 class ThisWillFail {
  public function foo() {
   return "Hello World!";
  }
 }
 $class = new ThisWillFail;
 $class->bar();
?>

  借助方法重载的帮助,代码能够捕获到这种调用且能够体面地给以处理。

  属性重载与方法重载差不多。这种情况下,类把读/写操作重定向(亦可称代理)到类的属性,这些属性在类中没有显式定义。这里的专门方法是__set()和__get()。依赖于错误报告等级,PHP翻译器通常在存取一个未定义的属性时,或者发出一个通知,或者推迟一下并潜在地定义这个变量。而如果使用属性重载,翻译器却可以在设置一个未定义的属性时调用__set(),而在存取一个未定义的属性值时调用__get()。
综上所述,利用重载技术可以实现在象用PHP这样的动态语言进行时软件开发时间的大大缩短。

  理论介绍至此,下面分析具体编码。

  三、持续性存储类举例

  下列代码,通过使用属性重载技术,用少于50行的PHP代码实现了上面所提到的持续性存储类。术语persistable意味着类可以从一个数据结构中描述一个元素,并保持与底端存储系统的同步。用编码的解释就是,外部代码可以使用类来实现从一个数据库表中选定一行。这样,在程序运行时,可以直接存取类的属性来操纵该行中的元素(读/取)。在脚本结束时,PHP将负责把更新的行数据回送到数据库中去。

  精心研读下面代码将有助于你理解什么是属性重载。

<?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:password@localhost/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