php教程:php设计模式介绍之策略模式(2)

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

例子

举例子说明,让我们做一个存储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()实现具体的测试过程。

(责任编辑:IT教学网)

更多