另外,如果你不喜欢该模式下对象与对象之间的连接方式,你可以更改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库有很大不同。
下文:《PHP设计模式介绍》第十章 规范模式
本文链接:http://www.blueidea.com/tech/program/2008/6029.asp
出处:phpchina
责任编辑:bluehearts
上一页 php设计模式介绍之观测模式 [3] 下一页
◎进入论坛网络编程版块参加讨论
|