php教程:php设计模式介绍之观测模式(4)

http://www.itjxue.com  2015-07-17 08:20  来源:未知  点击次数: 

另外,如果你不喜欢该模式下对象与对象之间的连接方式,你可以更改update()函数让它来发送一个信息(类似于本例中的错误信息数组或者几个信息对象)来避免引用自己。

这里是一个全新的ErrorHandler,不仅做了最新的修改并且还包含detach()函数:

class  ErrorHandler  {
var  $_observers=array();
var  $_error_info;
function  attach(&$observer)  {
$this->_observers[]  =&  $observer;
}
function  detach(&$observer)  {
foreach(array_keys($this->_observers)  as  $key)  {
if  ($this->_observers[$key]  ===  $observer)  {
unset($this->_observers[$key]);
return;
}
}
}
function  notify()  {
foreach(array_keys($this->_observers)  as  $key)  {
$observer  =&  $this->_observers[$key];
$observer->update($this);
}
}
function  getState()  {
return  $this->_error_info;
}
function  setState($info)  {
$this->_error_info  =  $info;
$this->notify();
}
}

你现在已经拥有了观测模式下的一个完整工具。

现在,回到本章的原始目标中,让我们看看如何在一个真正的PHP脚本中应用ErrorHandler。为了在一个PHP应用中包含观测者,你必须实例化ErrorHandler类,并确认函数set_error_handler()使用完全相同的参数。这听起来就像最近的一个问题:单件模式。

让我们作一个Factory()函数,它是一个简单的PHP函数,可以返回ErrorHandler的单态实例。

function  &getErrorHandlerInstance()  {
static  $instance  =  array();
if  (!$instance)  $instance[0]  =&  new  ErrorHandler();
return  $instance[0];
}

现在,让我们写一个错误记录句柄功能来获取单态ErrorHandler,改变它的状态来反映错误,并且通知“观测者”。


function  observer_error_handler(
$errno,  $errstr,  $errfile,  $errline,  $errcontext)  {
$eh  =&  getErrorHandlerInstance();
$eh->setState(array(
‘number’    =>  $errno

,’msg’        =>  $errstr
,’file’        =>  $errfile
,’line’        =>  $errline
,’context’  =>  $errcontext
));
}

也许你会注意到这里并没有ErrorHandler::notify()函数。为什么呢?因为ErrorHandler不论何时,只要状态一改变就会自动发出通知。

class  ErrorHandler  {
//  ...
function  setState($info)  {
$this->_error_info  =  $info;
$this->notify();
}
}

这种“默认通知”的方法,有利有弊。但先进之处在于客户端代码不需要包含通知的触发代码。

当然,如果主体对象的状态有好几处变化,所有的变动都对应不同的函数,你就可以选择让客体代码强制调用notify()函数。

自从你能正确使用这些辅助工具后,你给ErrorHandler添加的另一种类型的记录方式就会变得相当的容易?你现在只需要拥有向系统中写日志的权限。稍微查一下PHP手册(http://www.php.net/syslog),你就可以找到一些非常有用的函数来建立日志系统。这些可以很容易的被封装到一个新的类里,以便和ErrorHandler联合使用。


class  SyslogErrorLogger  {
function  SyslogErrorLogger($msg)  { define_syslog_variables(); openlog($msg,  LOG_ODELAY,  LOG_USER);
}

function  log($msg)  {
syslog(LOG_WARNING,  $msg);
}

function  update(&$error_handler)  {
$error  =  $error_handler->getState();
$this->log($error[‘msg’]);
}
}

注:错误日志的用处
日志是非常有用的――如果有人使用它们的话。但是,如果没有人使用日志,那么记录日志的代码就是一堆无用的代码
如果想知道更详细的评价,请查看
http://www.lastcraft.com/blog/index.php?p=4

结论

观测模式是非常有用的。这里的例子是完全静态的--观测者可以在脚本的初始化阶段被配置且被生成。要想展示观测模式的灵活性,最好是在一个更加动态的应用中--你需要根据脚本中的其他事情来添加或删除观测者。以常见的“生存时间”或者说该PHP脚本的允许执行时间打个比方,当同一个脚本在不同的情况下执行时,就可以根据不同的观测者分别配置,而不需要动态改变一个脚本的流程。这就和通过延长脚本执行时间的PHP-GTK库有很大不同。

(责任编辑:IT教学网)

更多