Head First设计模式阅读笔记(三)

状态模式:
概念:允许对象在内部状态改变时改变它的行为,对象看起来好像修改了它的类。

定义一个状态的父类,限定了可以执行的操作,并实例化用于存储现在状态。子类通过继承关系,实现不同状态下,进行各种操作的逻辑。这样可以保证对修改封闭,对添加开放。 当加入一个新的状态的时候,只需要继承父类并实现相应操作,并修改主线程逻辑即可。

同策略模式大同小异,但是实现的意图不一样。 策略模式是为了类的多样化,操作的多样化;状态模式是为了状态转换的易于实现和添加。

状态模式

 

代理模式:
概念:为另一个对象提供一个替身或占位符以访问这个对象

现在看来,应该就是类似于webservice或者是云服务吧。为客户提供一个接口,告诉他可以实现的功能,然后客户就可以使用这个接口完成想要进行的操作。实际上,这个接口只是调用了一个远程的服务,将数据传递过去并将结果传递回来。

现在的很多像soap和hessian都可以很好的完成,另外还有python和ruby等脚本语言应该也比较方便的实现。

代理模式

 

复合模式:

概念:复合模式结合两个或以上的模式,组成一个解决方案,解决一再发生的一般性问题。

问题的引出:

一大堆Quackable —– 策略模式,继承

有一只鹅出现了,它希望自己像一个Quackable —– 适配器模式

然后,呱呱叫学家决定要计算呱呱叫声的次数 —– 工厂模式生成

但是呱呱叫学家担心他们忘了加上QuackCounter装饰者 —– 装饰者模式

又是鸭子,又是鹅,又是Quackable的……我们有管理上的困扰 —– 组合模式和迭代器模式

当任何呱呱声响起时,呱呱叫学家都希望能被告知 —– 观察者模式

实际上,就MVC框架就是很好的一个复合模式

 

其他的一些模式

桥接模式:不只改变你的实现,也改变你的抽象。

生成器模式:封装一个产品的构造过程,并允许步骤构造。

责任链模式:让一个以上的对象有机会能够处理某个请求。(处理流程)

蝇量模式:让某个类的一个实例能用来提供许多“虚拟实例”。(类似不同大小不同内容的注释文字)

解释器模式:为语言创建解释器。(编译原理的词法分析,语法分析)

中介者模式:集中相关对象之间复杂的沟通和控制方式。(卖房中介,可以应对不同房源,不同买房主,卖房主,有自己的营销策略)

备忘录模式:可以让对象返回之前的状态。(动态规划里面的备忘录方法)

原型模式:创建给定类的实例的过程很昂贵或很复杂时使用。

访问者模式:可以为一个对象的组合增加新的能力,且封装并不重要时使用。

PHPUnit和xdebug

第二个月工作第12天,时间过得还蛮快地。

工作起来,就会找回期待晚上和周末的感觉。

说:白天越累,晚上越high,平时越忙,周末越high~

============================

今天使用PHPUnit和xdebug一起做了一下覆盖率测试,还蛮好玩的。

生成的那个html非常漂亮,把覆盖的代码全部都标记了,这样可以很快的发现那些分支,那些代码还没有被测试到。

安装方法网上都有,就不赘述了,我使用的是zend方法:

[zend_extension]
zend_extension=”/home/work/local/php/ext/xdebug.so”
;zend_extension=”/home/work/local/pikagent/lib/xdebug/xdebug_php5.so”
;zend_extension=”/home/work/local/pikagent/lib/xdebug/xdebug_php5.so”
xdebug.profiler_enable=on
xdebug.trace_output_dir=”/home/work/var/xdebug”
xdebug.profiler_output_dir=”/home/work/var/xdebug”
;xdebug.profiler_output_name=”script”

之后使用phpunit,就可以进行覆盖率测试了

命令:phpunit –coverage-html /home/work/var/xdebug TestCase.php  就可以了

其中 /home/work/var/xdebug 是存放结果的目录

TestCase.php就是单测文件

之后到xdebug的目录下,拷贝生成的html文件及相关css和image,打开就可以查看啦。

效果图: 上面是统计结果,绿色的话就是比较好,黄色的部分表示还没覆盖到,很直观啦。

测试覆盖率结果

Head First设计模式阅读笔记(二)

适配器模式:
概念:将一个类的接口,转换成客户期望的另一个接口。适配器让原本不兼容的类可以合作无间。
比如我们要将一个火鸡封装成一个鸭子,因为我们现在只有处理鸭子的机器方法。那么我们使用一个将火鸡转换成鸭子的适配器类,类中包含一个火鸡对象,当需要它鸣叫时,调用火鸡的鸣叫方法,需要飞时,调用火鸡的飞行方法。通俗理解就像插头转换器一样,所以叫适配器模式。

与装饰者模式(不改变接口,但加入方法)不同的是,适配器模式是直接将一个接口转成另一个接口。

适配器模式

外观模式:

概念:提供了一个统一的接口,用来访问子系统中的一群接口。外观定义了一个高层接口,让子系统更容易使用。

就是说把一些相关的类的操作集合在了一起,创建了一个流程的感觉。类似于功能的集合,比如按遥控器的家庭影院按钮,会关灯,开音响,下降幕布等一系列操作。

外观模式

模板方法模式:

概念:在一个方法中定义一个算法的骨架,而将一些步骤延迟到子类中。模板方法使得子类可以在不改变算法结构的情况下,重新定义算法中的某些步骤。

算法的骨架使用final定义,不可改变。可以延迟的步骤使用抽象方法abstract。比如可以定义制造咖啡,牛奶和茶的方法步骤,只是冲泡的内容不一样,添加的调料不一样,都需要烧开水和装杯(可能杯子也不一样)。

模板方法模式

迭代器模式:

概念:提供一种方法顺序访问一个聚合对象中的各个元素,而又不暴露其内部的表示。

比较好理解,就是比如有几种集合元素,数组,ArrayList,Hashmap,使用Iterator迭代获得全部元素。需要注意的是可以自己写一些Iterator方法,和compareTo方法,来完成比较。

迭代器模式

组合模式:

概念:允许你将对象组合成树形结构来表现“整体/部分”层次结构。组合能让客户以一致的方式处理个别对象以及对象组合。

就是借鉴的树的概念,一个结点,可以是父节点,也可以是子结点,迭代器会相同对待。他们共同继承自Component,可以将对象的集合以及个别的对象一视同仁。。复杂的是在迭代的过程中,要考虑到遍历的顺序,可以参考深搜和广搜。

组合模式

 

PHPUnit学习

来自百度百科……
“PHPUnit是一个轻量级的PHP测试框架。它是在PHP5下面对JUnit3系列版本的完整移植,是xUnit测试框架家族的一员(它们都基于模式先锋Kent Beck的设计)。
单元测试是几个现代敏捷开发方法的基础,使得PHPUnit成为许多大型PHP项目的关键工具。这个工具也可以被Xdebug扩展用来生成代码覆盖率报告 ,并且可以与phing集成来自动测试,最后它还可以和Selenium整合来完成大型的自动化集成测试。”

简单来讲,使用PHPUnit的目的,就是在测试代码的时候,设置好检测点,检测方法和结果比对方式,进行大量的自动的代码测试。以下面这个例子说明:
assertEquals(0, sizeof($fixture));
   }

   public function testArrayContainsAnElement() {
   /* Create the Array fixture*/
   $fixture = array();
   /*Add an element to the Array * fixture*/
   $fixture[] = ‘Element’;
   /*Assert that the size of the * Array fixture is 1*/
   $this->assertEquals(1, sizeof($fixture));
   }
  }
?>
里面定义了一个测试点,测试Array的功能,两个测试方法:为空和为一个元素的时候;结果比对就是数组的大小,为空=0,一个元素=1. 这样使用phpunit ArrayTest就可以进行自动的测试,并给出测试结果。

要点:
  1 编写的测试用例是一个php脚本
  2 require_once ‘PHPUnit/Framework.php’是必须的,另外,你需要在测
  试用例脚本中包含你需要测试的代码
  3 测试用例的主体必须写在类中,类名必须和文件名保持一致,还必须是
  PHPUnit_Framework_TestCase的子类
  4 每一个测试用例都是一个public的成员函数,必须以test开头
  5 程序的输出使用assert*系列函数来进行验证
  该用例需要在Shell下键入phpunit ArrayTest.php来运行。结果如下示:
  [username@machine xx]$ phpunit ArrayTest.php
  PHPUnit 3.1.3 by Sebastian Bergmann.
  ..
  Time: 0 seconds
  OK (2 tests)
  结果中点号代表一个用例通过(即assert系列函数都通过)。如果将上面的测试用例testArrayContainsAnElement用例assertEquals函数中的1改为0,则运行结果为:
  PHPUnit 3.1.3 by Sebastian Bergmann.
  .F
  Time: 0 seconds
  There was 1 failure:
  1) testArrayContainsAnElement(ArrayTest)
  Failed asserting that matches expected value .
  /home/wiki/apache/htdocs1.5.0/ArrayTest.php:29
  FAILURES!
  Tests: 2, Failures: 1.
  可以很明显看出结果由原来的 .. 变为 .F ,F表示第二个测试用例未通
  过,并且具体与哪一条验证不符都在后有详细说明。
  除了F外,一个测试用例还有I(未完成),S (跳过),E (错误)三种状态

高级的一些用法比如:依赖关系和数据提供者大家可以百度学习,其实也不太用到
还有一些比如测试异常的方式,可以用:
try catch + fail法
public function testException() {
try {
// 这里写上会引发异常的代码
} catch (InvalidArgumentException $expected) {
// 抓到异常测试通过
return ;
}

// 没抓到异常就算失败
$this->fail(‘An expected exception has not been raised.’ );
}

常用的assert****
assertEquals(array $expected, array $actual[, string $message = ”])
断言数组$actual和数组$expected相同
assertFalse(bool $condition[, string $message = ”])
断言$condition的结果为false
assertTrue(bool $condition[, string $message = ”])
断言$condition为True,否则就报告错误

其他的话,其实都可以用assertEquals替代来写,实际应用实际操作吧。
其实做测试主要还是设计的考虑和边界值的选取等。

Mac OS Lion系统下安装apache,php,pear和phpunit

八月上班的第一天,加入项目组了,准备开始学习测试
============================
使用MacOS Lion还是不错的,想着做开发应该没啥问题吧,php嘛。但是实际操作起来还是有很多不同的。
首先他自带了php 5.3.6,版本很新啊,和apache2,所以就不需要装了
在命令行下直接php命令就可以了,但是如果要打开apache的功能,还需要做一些操作
开启apache对php的支持
在/etc/apache2/httpd.conf中,打开LoadModule php5_module libexec/apache2/libphp5.so这一项
然后去系统偏好设置,共享中打开web共享,然后访问http://localhost可以看到久违的It works!
自己设置的网站是在 ~/Sites/文件夹下,对应的url是
http://localhost/~[your_user_name]/index.php
可以使用常用的phpinfo()来试一下开通了没有。

然后就要开始装pear了
cd /usr/local下,然后使用
curl http://pear.php.net/go-pear | sudo php
命令,发现php版本太新了,提示用go-pear.phar > go-pear.php文件才行
安装完成如果发现pear命令可以用了,那就说明好了,如果还不行的话,就百度一下吧⋯⋯

开始安装phpunit
首先升级pear,使用 pear upgrade pear
然后添加pear的频道
pear channel-discover components.ez.no

pear channel-discover pear.phpunit.de

pear channel-discover pear.symfony-project.com

之后使用
pear install –alldeps –force phpunit/phpunit
即可完成安装
安装完成如果运气好的话,比如我在xp下面,就可以直接使用phpunit命令了,
但是在mac下还是不行,提示
PHP Warning: require_once(PHP/CodeCoverage/Filter.php): failed to open stream:
No such file or directory in phpunit on line 38
我先用pear install phpunit/PHP_CodeCoverage命令装了,还是不行。
肯定就是路径问题,可以使用修改include_path的方法,但是不太明白,所以
打开pear目录下phpunit在

Head First设计模式

这是一本好书
上班第17天,今天发工资,哇咔咔
============================================
策略模式:
概念:定义了算法族,分别封装起来,让它们之间可以互相替换,此模式让算法的变化独立于使用算法的客户。
通俗来讲,就是有一个父类型,他有一些方法,子类型可以继承这些方法,但是这些方法在子类中的表现又不尽相同。
所以将父类型中的一些方法,也变成接口和类来处理,这样可以在构造子类的时候,选择对应的父类不同的方法来构造。

策略模式

观察者模式:
概念:定义了对象之间的一对多依赖,这样依赖,当一个对象改变状态时,它的所有依赖者都会收到通知并自动更新。
例子是天气预报,接收到新数据后对三个指示器进行更新。指示器继承自object并都有update方法。主题对象将不同的object加入到array,在获取数据后对全部object进行自动更新。

观察者模式

装饰者模式:
概念:动态地将责任附加到对象上。若要扩展功能,装饰者提供了比继承更有弹性的替代方案。
比如星巴兹咖啡的解决方案:先定义一个基类,有描述和花费两个方法,一般的咖啡直接继承该父类即可。如果需要加调料,使用继承自基类的装饰者类,用原始的对象构造新的对象,并添加额外的花费即可。

装饰者模式

工厂方法模式:
概念:定义了一个创建对象的接口,但由子类决定要实例化的类是哪一个。工厂方法让类把实例化推迟到子类。
在制作pizza的时候,使用pizza工厂类,来根据不同的type选择实例化不同的pizza style。这样可以保证在对之后的处理独立。
也就是把一串的if else移到专门的工厂类里面去,保证代码的稳定性。
包括:一般工厂模式,抽象工厂模式(可以有很多不同的工厂实现自同一个抽象工厂,比如pizza的原料也可以不同)

工厂模式

单件模式:
概念:确保一个类只有一个实例,并提供一个全局访问点
Singleton模式的实现基于两个要点:
  1)不直接用类的构造函数,而另外提供一个Public的静态方法来构造类的实例。通常这个方法取名为Instance。Public保证了它的全局可见性,静态方法保证了不会创建出多余的实例。
  2)将类的构造函数设为Private,即将构造函数”隐藏”起来,任何企图使用构造函数创建实例的方法都将报错。这样就阻止了开发人员绕过上面的Instance方法直接创建类的实例。
  通过以上两点就可以完全控制类的创建:无论有多少地方需要用到这个类,它们访问的都是类的唯一生成的那个实例。

命令模式:
概念:将“请求”封装成对象,以便使用不同的请求、队列或者日志来参数化其他对象。命令模式也支持可撤销的操作。
定义一个命令接口,具有execute方法,其他命令已此为基础。使用命令的对象包含一个command,通过set方法指定不同的命令并调用execute方法即可实现命令操作。

命令模式

蟒蛇python

上班第16天,明天就发工资咯。也到月末了,可以休息一下了。
把仙剑5赶紧通关,然后弄点好吃的。
利用空闲时间学习一点python,名字还是很霸气的
=========================================
1.变量可以随便用的,也可以用global
2.def就是定义函数
3.class就是定义类
4.字符串很方便
5.无类型之说
6.while可以有else
7.没有switch
8.__init__方法类似于C++、C#和Java中的 constructor
9.不需要分号,最好一行一句逻辑语句
10.self就是this
11.类,if,def等后面都要有冒号
12.注意缩进格式,不能随便缩进
13.继承就是class ChildClass(FatherClass)
14.模块使用import引入

PHP调试时用print_r()和var_dump()的区别

上班第14天,开始为用户中心测试后台添加实际功能了。
列车追尾了,很痛心。
北京连续下雨,像疯了一样。
好好生活,珍惜生命。

==================以下是转的=============================================

print_r()结构智能地显示传递给它的东西,而不像echo 和print()那样将所有的东西都转换成字符串.字符串和数字都被简单地输出.数组以括起来的键和值的列表形式表示.以Array开头:

$arr = array(‘name’=> ‘bob’,’age’ => 20, ‘sex’ => ‘man’);

print_r($arr);

输出形如:

Array
{
[name] => bob,

[age] => 20,

[sex] => man
}

但对,用print_r()来显示布尔值和NULL是没有意义的:

如:

print_r(true); //输出1

print_r(false); //没有输出

print_r(null); //没有输出

因此,调试的时候用var_dump()比print_r()更好.var_dump()函数用适合阅读的格式显示任何PHP值:

var_dump(true); //输出 bool(true)

var_dump(false); // bool(false)

var_dump(null); // bool(null)

var_dump(array(‘name’ => ‘bob’, ‘age’ => 20));

array(2) {
[‘name’] => string(3) ‘bob’
[‘age’] => int(20)
}

正则表达式

今天周五啦,事情不多,学习学习正则表达式。昨天把Lion装完了,体验还是不错的嘛。
继续学习。。。
=================昏哥线===================
正则表达式用于字符串处理、表单验证等场合,实用高效。现将一些常用的表达式收集于此,以备不时之需。

匹配中文字符的正则表达式: [u4e00-u9fa5]
评注:匹配中文还真是个头疼的事,有了这个表达式就好办了

匹配双字节字符(包括汉字在内):[^x00-xff]
评注:可以用来计算字符串的长度(一个双字节字符长度计2,ASCII字符计1)

匹配空白行的正则表达式:ns*r
评注:可以用来删除空白行

匹配HTML标记的正则表达式:<(S*?)[^>]*>.*?|<.*? />
评注:网上流传的版本太糟糕,上面这个也仅仅能匹配部分,对于复杂的嵌套标记依旧无能为力

匹配首尾空白字符的正则表达式:^s*|s*$
评注:可以用来删除行首行尾的空白字符(包括空格、制表符、换页符等等),非常有用的表达式

匹配Email地址的正则表达式:w+([-+.]w+)*@w+([-.]w+)*.w+([-.]w+)*
评注:表单验证时很实用

匹配网址URL的正则表达式:[a-zA-z]+://[^s]*
评注:网上流传的版本功能很有限,上面这个基本可以满足需求

匹配帐号是否合法(字母开头,允许5-16字节,允许字母数字下划线):^[a-zA-Z][a-zA-Z0-9_]{4,15}$
评注:表单验证时很实用

匹配国内电话号码:d{3}-d{8}|d{4}-d{7}
评注:匹配形式如 0511-4405222 或 021-87888822

匹配腾讯QQ号:[1-9][0-9]{4,}
评注:腾讯QQ号从10000开始

匹配中国邮政编码:[1-9]d{5}(?!d)
评注:中国邮政编码为6位数字

匹配身份证:d{15}|d{18}
评注:中国的身份证为15位或18位

匹配ip地址:d+.d+.d+.d+
评注:提取ip地址时有用

匹配特定数字:
^[1-9]d*$    //匹配正整数
^-[1-9]d*$   //匹配负整数
^-?[1-9]d*$   //匹配整数
^[1-9]d*|0$  //匹配非负整数(正整数 + 0)
^-[1-9]d*|0$   //匹配非正整数(负整数 + 0)
^[1-9]d*.d*|0.d*[1-9]d*$   //匹配正浮点数
^-([1-9]d*.d*|0.d*[1-9]d*)$  //匹配负浮点数
^-?([1-9]d*.d*|0.d*[1-9]d*|0?.0+|0)$  //匹配浮点数
^[1-9]d*.d*|0.d*[1-9]d*|0?.0+|0$   //匹配非负浮点数(正浮点数 + 0)
^(-([1-9]d*.d*|0.d*[1-9]d*))|0?.0+|0$  //匹配非正浮点数(负浮点数 + 0)
评注:处理大量数据时有用,具体应用时注意修正

匹配特定字符串:
^[A-Za-z]+$  //匹配由26个英文字母组成的字符串
^[A-Z]+$  //匹配由26个英文字母的大写组成的字符串
^[a-z]+$  //匹配由26个英文字母的小写组成的字符串
^[A-Za-z0-9]+$  //匹配由数字和26个英文字母组成的字符串
^w+$  //匹配由数字、26个英文字母或者下划线组成的字符串
评注:最基本也是最常用的一些表达式

原载地址:http://lifesinger.3322.org/myblog/?p=185