php教程:php设计模式介绍之策略模式(2)
例子
举例子说明,让我们做一个存储PHP参数的cache。这个cahce类需要把变量以PHP识别的方式写入到一个文件当中,所以你可以在以后加载该文件并使用它。这个类还应该可以让你为每个数据加个标识符和存储的方式。
数据缓存
注:缓存是为了在接下来的操作中继续使用而对资源进行缓存。你可以通过建立和使用缓存来节省直接从原数据库获取数据的时间。这方面的例子最常见的就是访问数据库或者解析大的XML文档,或者大的配置文件。
缓存也会出现一个问题:你的缓存可能会失去与原数据的同步。或者缓存需要使用太多内存。
最开始,我们开发一个缓存操作,并不使用策略模式。
因为你可能需要缓存的不止一个值,所以你需要使用标识符来标识出你需要指定的元素。在这个例子中,标识符就是’application_config’。下面试一个如果使用cache的例子。
// PHP4
$config_cache =& new VarCache(‘application_config’);
if ($config_cache->isValid()) {
$config = $config_cache->get();
} else {
$config = slow_expensive_function_to_get_config();
$config_cache->set($config);
}
这个代码生成了一个新的VarCache对象存放在$config_cache变量里面。这个数据在缓存中的标识符是 ‘application_config’。如果在缓存里面有这个数据, isValid() 将返回真( true )并且获取缓存中的数据。反之,值被重新获取并写入缓存当中,以便下次使用。
按照一般的需求,让我们开始编写这段代码来进行测试。首先,如果缓存中没有该数据, isValid() 方式函数应该返回非值(false)。
class VarCacheTestCase extends UnitTestCase {
function TestUnsetValueIsInvalid() {
$cache =& new VarCache(‘foo’);
$this->assertFalse($cache->isValid());
}
因为VarCache现在没有代码,所以最简单的方式就是先构造一个方式函数。
class VarCache {
function isValid() {}
}
这样,我们就可以继续了。
class VarCacheTestCase extends UnitTestCase {
function TestUnsetValueIsInvalid() { /* ... */ }
function TestIsValidTrueAfterSet() {
$cache =& new VarCache(‘foo’);
$cache->set(‘bar’);
$this->assertTrue($cache->isValid());
}
上面的测试校验了缓存的数据是否是可用的。
开始编写cache类的主要部分。VarCache 引入一个标识符, 所以constructor了一个应该记录它的对象实例。这里面还有一个set()的方式函数,用来把数据存入缓存,或者当数据存在时,修改缓存当中的数据。
class VarCache {
var $_name;
function VarCache($name) {
$this->_name = ‘cache/’.$name;
}
function isValid() {
return file_exists($this->_name.’.php’);
}
function set() {
$file_handle = fopen($this->_name.’.php’, ‘w’);
fclose($file_handle);
}
}
对象实例的参数$_name 存放了缓存的标识符。在这个简单的操作中, $_name 被用来生成文件名(在实际的使用可能会数据库或者其它的数据源代替) set() 使用 fopen() 和 fclose() 来 “访问” 基于$_name的文件。当调用set()后, file_exists()在VarCache::isValid()里面调用返回真(true)。
运行这个测试来产生一个我们预期的结果;但是实际情况是报错!为什么呢?第一次运新的时候没有生成文件,所以第二次运行的时候找不到文件,显然我们不希望这种情况出现。我们期望的是每一次运行代码都是互不影响的。
幸运的是,把总体测试框架和特定功能的简单测试结合起来,我们就可以得到灵活的测试环境,并且在以后的测试中方便地使用。UnitTestCase::setUp()实现框架的初始化,而UnitTestCase::tearDown()实现具体的测试过程。