php设计模式介绍之注册模式(5)

2011-11-16
浏览
导读:现在你已经看到在PHP5中的静态调用接口,下面让我们在PHP4中实现相同的接口。在前 面的PHP4静态类变量部分,实现需要使用函数静态变量返回引用来跟踪 。PHP4版本的静态调用接口测试与PHP5版本的测试类似。 代码: /

现在你已经看到在PHP5中的静态调用接口,下面让我们在PHP4中实现相同的接口。在前 面的PHP4“静态类变量”部分,实现需要使用“函数静态变量返回引用”来跟踪 。PHP4版本的静态调用接口测试与PHP5版本的测试类似。

代码:

//  PHP4

class RegistryStaticPHP4TestCase extends UnitTestCase {
function  testRegistryStatic() {
$this->assertFalse(RegistryStatic::isValid (‘key’));
$this->assertNull(RegistryStatic::get(‘key’));
$test_value = ‘something’; RegistryStatic::set(‘key’,  $test_value);
$this->assertReference($test_value, RegistryStatic::get (‘key’));
}
}

以下是符合测试要求的代码实现。

代码 :

class RegistryStatic {
function &_getRegistry() { static  $store = array(); return $store;
}
function isValid($key) {
$store  =& RegistryStatic::_getRegistry();
return array_key_exists($key, $store);
}
function &get($key) {
$store =& RegistryStatic::_getRegistry ();
if (array_key_exists($key, $store))
return $store[$key];
}
function set($key, &$obj) {
$store =& RegistryStatic::_getRegistry ();
$store[$key] =& $obj;
}
}

这个实现方法的重点是 getRegistry()方法返回一个对静态数组的引用。

$store =&  RegistryStatic::_getRegistry();这一行,在随后的函数中把变量$store通过引用赋给静态数组,允许 所有的函数可以静态访问数组,允许所有的方法可以被静态调用。

也可以不使用PHP4“静 态类变量跟踪”达到相同的效果:将原先的基于单件模式的Registry类与一个包装类结合以达到允 许静态调用。这个类与testRegistryStatic()有相同的测试代码,但是他的实现如下所示:

代码 :

class RegistryStatic {
function isValid($key) {
$reg =&  Registry::getInstance();
return $reg->isValid($key);
}
function  &get($key) {
$reg =& Registry::getInstance();
return $reg->get ($key);
}
function set($key, &$obj) {
$reg =&  Registry::getInstance();
$reg->set($key, $obj);
}
}

结论:

虽然注册模式简化了对大量对象的访问,但是仍然有许多问题——与全局变量联合。 你需要确定要求的属性Key在访问之已经被初始化了,而且设置属性的方法可以全局访问,你的对象仍然 可能在你的代码的其他部分出乎意料的被替换掉。显然,全局数据非常有好处,方便,但是你需要时刻 记住任何全局数据都是有一些不安全的。

内嵌的ReGIStry模式

除了单独使用注册模式 ——如本章所示,Registry模式与其他对象结合时功能也是非常强大。例如:当对象的创建 代价非常昂贵(例如需要查询大量数据库来初始化对象)时,而且对象在这个应用中被使用一次或多次 ,如果这样,你能创建一个结合了工作模式 (见第三章) 和注册模式 的“Finder”类以获 得已经创建的对象的缓存而不用再次创建他们?

以下是一个Contact类,AddressBook类是工厂类 。

代码:

class AddressBook {
function &findById($id) {
return new Contact($id);
}
}
class Contact {
function Contact ($id) {
// expensive queries to create object using $id
}
// ...  other methods
}

你可以在AddressBook类中插入Registry模式来提供缓存。代码 可以如下所示:

class AddressBook {
var $registry;
function  AddressBook() {
$this->registry =& Registry::getInstance();
}
function &findById($id) {
if (!$this->registry->isValid($id)) {
$this->registry->set($id, new Contact($id));
}
return $this- >registry->get($id);
}
}

AddressBook类的构造函数将registry绑定 到一个实例变量。当创建了一个特定的ID并被findById()方法调用时,Registry被检查以确定对象是否 已经被缓存。如果没有,将创建一个新的对象并存储在Registry中。被调用的对象将通过函数从 Registry中取出并被返回。

thinkphp 3.2框架视图模型 实例视图查询结

PHP实现微信小程序在线支付功能(代码实例)

Yii1.0 不同页面多个验证码的使用实现

详解laravel中blade模板带条件分页

PHP函数参数传递的方式整理