上一篇我们介绍了设计模式的特性并且详细讲解了4种创建型模式,创建型模式是负责如何产生对象实例的,现在我们继续来给大家介绍结构型模式。
一、什么是结构型模式? 结构型模式是解析类和对象的内部结构和外部组合,通过优化程序结构解决模块之间的耦合问题。
二、结构型模式的种类: 适配器模式 桥接模式 装饰模式 组合模式 外观模式 享元模式 代理模式
1、 适配器模式(Adapter) 将一个类的接口转换成客户希望的另一个接口,适配器模式使得原本的由于接口不兼容而不能一起工作的那些类可以一起工作。 应用场景:老代码接口不适应新的接口需求,或者代码很多很乱不便于继续修改,或者使用第三方类库。 代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
|
/** * 优才网公开课示例代码
*
* 适配器模式
*
* @author 优才网全栈工程师教研组
* @see http://www.ucai.cn
*/
//老的代码 class User {
private $name ;
function __construct( $name ) {
$this ->name = $name ;
}
public function getName() {
return $this ->name;
}
} //新代码,开放平台标准接口 interface UserInterface {
function getUserName();
} class UserInfo implements UserInterface {
protected $user ;
function __construct( $user ) {
$this ->user = $user ;
}
public function getUserName() {
return $this ->user->getName();
}
} $olduser = new User( '张三' );
echo $olduser ->getName(). "n" ;
$newuser = new UserInfo( $olduser );
echo $newuser ->getUserName(). "n" ;
|
注意点:这里的新接口使用了组合方式,UserInfo内部有一个成员变量保存老接口User对象,模块之间是松耦合的,这种结构其实就是组合模式。不要使用继承,虽然UserInfo继承User也能达到同样的目的,但是耦合度高,相互产生影响。
2、 桥接模式 将抽象部分与它的实现部分分离,使它们都可以独立变化 特点:独立存在,扩展性强
应用:需要不断更换调用对象却执行相同的调用方法,实现扩展功能
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
|
/** * 优才网公开课示例代码
*
* 桥接模式
*
* @author 优才网全栈工程师教研组
* @see http://www.ucai.cn
*/
abstract class Person {
abstract function getJob();
} class Student extends Person {
public function getJob() {
return '学生' ;
}
} class Teacher extends Person {
public function getJob() {
return '老师' ;
}
} class BridgeObj {
protected $_person ;
public function setPerson( $person ) {
$this ->_person = $person ;
}
public function getJob() {
return $this ->_person->getJob();
}
} $obj = new BridgeObj();
$obj ->setPerson( new Student());
printf( "本次桥接对象:%sn" , $obj ->getJob());
$obj ->setPerson( new Teacher());
printf( "本次桥接对象:%sn" , $obj ->getJob());
|
3、 装饰模式 动态地给一个对象添加额外的职责。在原有的基础上进行功能增强。 特点:用来增强原有对象功能,依附于原有对象。 应用:用于需要对原有对象增加功能而不是完全覆盖的时候
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
|
/** * 优才网公开课示例代码
*
* 装饰模式
*
* @author 优才网全栈工程师教研组
* @see http://www.ucai.cn
*/
//产品 abstract class Person {
abstract function getPermission();
} //被装饰者 class User extends Person {
public function getPermission() {
return '公开文档' ;
}
} //装饰类 class PermUser extends Person {
protected $_user ;
protected $_special = '' ;
function __construct( $user ) {
$this ->_user = $user ;
}
public function getPermission() {
return $this ->_user->getPermission() . $this ->_special;
}
} //装饰类产品 class JavaUser extends PermUser {
protected $_special = ' java程序' ;
} class CPlusUser extends PermUser {
protected $_special = ' c++程序' ;
} $user = new User();
printf( "permission:%sn" , $user ->getPermission());
$user = new JavaUser( $user );
printf( "permission:%sn" , $user ->getPermission());
$user = new CPlusUser( $user );
printf( "permission:%sn" , $user ->getPermission());
|
大家想想装饰和继承的区别在哪? 如果是上面的例子,如果用继承,是CPlusUser继承JavaUser还是反过来呢?谁也不知道最终使用者需要哪一种。
在多层关系的情况下,装饰是和顺序无关并且随时增加装饰,而继承只能是特定的顺序,所以装饰模式会更加的灵活。
4、组合模式 将对象组合成树形结构表示“部分-整体”的层次结构。 特点:灵活性强 应用:对象的部分-整体的层次结构,模糊组合对象和简单对象处理问题代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
|
/** * 优才网公开课示例代码
*
* 组合模式
*
* @author 优才网全栈工程师教研组
* @see http://www.ucai.cn
*/
//继承模式 class UserBaseInfo {
private $name ;
function __construct( $name ) {
$this ->name = $name ;
}
public function getName() {
return $this ->name;
}
} class User extends UserBaseInfo {
private $login = false;
public function setLogin( $islogin ) {
$this ->login = $islogin ;
}
public function isLogin() {
return $this ->login;
}
} $user = new User( '张三' );
$user ->setLogin(true);
if ( $user ->isLogin()) {
echo $user ->getName(). "已经登录了n" ;
} else {
echo $user ->getName(). "还没有登录n" ;
} //组合模式 class LoginInfo {
protected $user ;
protected $login = false;
public function setLogin( $user , $isLogin ) {
$this ->user = $user ;
$this ->login = $isLogin ;
}
public function isLogin() {
return $this ->login;
}
} $user = new User( '张三' );
$login = new LoginInfo();
$login ->setLogin( $user , true);
if ( $login ->isLogin()) {
echo $user ->getName(). "已经登录了n" ;
} else {
echo $user ->getName(). "还没有登录n" ;
} //部分可以更换,用继承则不行 class Admin {
protected $level ;
function __construct( $level ) {
$this ->level = $level ;
}
function getLevel() {
return $this ->level;
}
} $admin = new Admin(1);
$login ->setLogin( $admin , true);
if ( $login ->isLogin()) {
printf( "级别为 %d 的管理员已经登录了n" , $admin ->getLevel());
} else {
printf( "级别为 %d 的管理员还没有登录n" , $admin ->getLevel());
} |
上面的例子分别展示了使用继承和组合来处理新功能,在简单的情况下看似区别不大,但在项目后期越来越复杂的时候组合模式的优势就越来越明显了。 例如上面的登录信息,如果要增加登录次数、最后登录时间、登录ip等信息,登录本身就会变成一个比较复杂的对象。如果以后有新的需求比如好友信息、用户的 访问信息等,再要继承的话,用户类就会变得非常庞大,难免各父类之间没有冲突的变量和方法,而外部访问用户类的众多方法也变得很费劲。采用组合模式后,一 个类负责一个角色,功能区分非常明显,扩展方便。
5、 外观模式(门面模式) 为了系统中的一组接口提供一个一致的界面 特点:向上抽取,有共性 应用:内部接口众多,由统一的接口来调用
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
|
/** * 优才网公开课示例代码
*
* 外观模式,也叫门面模式
*
* @author 优才网全栈工程师教研组
* @see http://www.ucai.cn
*/
class Operation {
public function testPlus() {
printf( "plus: %sn" , (1 + 2 == 3 ? 'true' : 'false' ));
}
public function testMinus() {
printf( "minus: %sn" , (3 - 2 == 2 ? 'true' : 'false' ));
}
public function testTimes() {
printf( "times: %sn" , (2 * 3 == 6 ? 'true' : 'false' ));
}
} class Tester {
protected $_operation ;
function __construct() {
$this ->_operation = new Operation();
}
public function testAll() {
$this ->_operation->testPlus();
$this ->_operation->testMinus();
$this ->_operation->testTimes();
}
} //测试用例,测试全部接口 $tester = new Tester();
$tester ->testAll();
|
门面模式估计大家在实际代码中都已经使用到了,接口较多时把相似功能的接口封装成一个接口供外部调用,这就是门面模式。
6、 享元模式 运用共享技术有效地支持大量细粒度对象,采用一个共享来避免大量有相同内容对象的开销。这种开销中最直观的就是内存的损耗。 特点:高效性,共享性 应用:系统底层的设计。例如字符串的创建。如果两个字符串相同,则不会创建第二个字符串,而是第二个的引用直接指向第一个字符 串。$str1=”abc”,$str2=”abc”.则内存存储中只会创建一个字符串“abc”而引用$str1.$str2都会指向它。
7、 代理模式 为其他对象提供一个代理来控制对这个对象的访问,就是给某一对象提供代理对象,并由代理对象控制具体对象的引用。能够协调调用者和被调用者,能够在一定程度上降低系统的耦合性。 特点:低耦合性,独立性好,安全性 应用:客户访问不到或者被访问者希望隐藏自己,所以通过代理来访问自己。
代码实现
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
|
/** * 优才网公开课示例代码
*
* 代理模式
*
* @author 优才网全栈工程师教研组
* @see http://www.ucai.cn
*/
//内部对象 class User {
public function getName() {
return '张三' ;
}
public function getType () {
return '付费用户' ;
}
} //代理接口定义,例如开放平台 interface UserInterface {
function getName();
} //代理对象 class UserProxy implements UserInterface {
protected $_user ;
function __construct() {
$this ->_user = new User();
}
public function getName() {
return $this ->_user->getName();
}
} //内部调用 $user = new User();
printf( "user name:%sn" , $user ->getName());
printf( "user type:%sn" , $user -> getType ());
//外部调用 // $user = new UserProxy(); // printf("user name:%sn", $user->getName()); // printf("user type:%sn", $user->getType()); //不能访问,及时知道内部对象有这个方法 |
三、总结 1、代理模式、适配器模式、门面模式、装饰模式的区别 a、 相同之处:都封装一个内部对象,调用内部对象的方法 b、 不同之处:各自有各自的特性和应用场景,不能相互替代。所以用的时候要仔细分析用那种合适。
2、 关于模式的选用问题 模式的选用要根据实际的业务需求,通过对业务逻辑的仔细分析,再根据模式具有的特性和应用场景进行合理的选择和区分。大部分情况下业务的场景决定了哪种模 式,而不是选择哪个模式去实现一个业务,少数情况几种模式确实都能解决问题,那主要就是考虑以后的扩展了。 到这里我们已经了解了7种结构型模式,下一篇我们继续给大家介绍设计模式的行为型模式,先预览一下行为型模式的种类吧:
u模版方法模式
u命令模式
u迭代器模式
u观察者模式
u终结者模式
u备忘录模式
u解释器模式
u状态模式
u策略模式
u职责链模式
u访问者模式
优才网公开课:http://www.ucai.cn/train/
转载自:http://www.ucai.cn/blogdetail/7020?action=gt&mid=1
相关推荐
《Learning PHP设计模式》中文翻译版2014年第一版第一次印刷中国电力出版社出版,不是扫描版,纯pdf版带目录结构清晰度为无障碍阅读。
php设计模式 运行测试./vendor/bin/phpunit 创建型模式实例 单例模式 :check_mark_button: 工厂模式 :check_mark_button: 抽象工厂模式 :check_mark_button: 简单工厂模式 :check_mark_button: 静态工厂模式 :check_...
通过学习如何在代码中使用设计模式,可以更高效...第5部分 MySQL和PHP设计模式 第11章 通用类负责连接,代理模式保证安全 第12章 策略设计模式的灵活性 第13章 职责链设计模式 第14章 利用观察者模式构建多设备CMS
mvc设计模式的讲解与使用 常见WEB开发语言的MVC框架 jsp+servlet+javaBean实现MVC –流程 jsp+servlet+javaBean实现MVC –技术点 PHP与JAVA在MVC模式上的不同之处 MVC设计模式的优势 MVC设计模式的缺点 MVC结构提供...
本文实例讲述了PHP设计模式:外观模式Facade。分享给大家供大家参考,具体如下: 1. 概述 外观模式,我们通过外观的包装,使应用程序只能看到外观对象,而不会看到具体的细节对象,这样无疑会降低应用程序的复杂度...
结构模式之间有许多相似之处,因为在对象模型结构中存在几种关系:类之间的继承和实现,加上对象组合,这些是PHP设计模式的主要目的。 对象组合的最大优势在于可以在运行时基于配置建立起一个关系,并不用修改实际的...
PHP设计模式 通过简单的示例学习OOP机制... 对于波兰社区,我还在博客上写了更多有关模式的内容: 创作性: 结构: 行为的: 模板方法 我还向您推荐两个具有设计模式的出色网站: 我见过(必须看到)的有关该主题...
本文实例讲述了PHP设计模式:组合模式Composite。分享给大家供大家参考,具体如下: 1. 概述 在数据结构里面,树结构是很重要,我们可以把树的结构应用到设计模式里面。 例子1:就是多级树形菜单。 例子2:文件和...
设计模式可以看作是一种计算机编程的结构化方法,介于编程范式和具体算法之间。 实施清单 贡献 始终欢迎捐款。 语言无所谓。 只要确保您正在实现设计模式即可。 欢迎公关。 要开始开发,请遵循以下结构: 类别/...
本文实例讲述了PHP设计模式之装饰器模式定义与用法。分享给大家供大家参考,具体如下: 什么是装饰器模式 作为一种结构型模式, 装饰器(Decorator)模式就是对一个已有结构增加”装饰”. 适配器模式, 是为现在有结构...
本文实例讲述了PHP设计模式之模板方法模式。分享给大家供大家参考,具体如下: 定义一个操作中的算法的框架,而将一些步骤延迟到子类中。使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤。 模板...
二、工厂模式结构图 三、工厂模式中主要角色 抽象产品(Product)角色:具体产品对象共有的父类或接口 具体产品(Concrete Product)角色:实现抽象产品角色所定义的接口,并且工厂方法模式所创建的每一个对象都是某...
PHP100视频教程58:快速设计一个企业网站之BIWEB PHP100视频教程59:关于BIWEB常见问题和结构分析 PHP100视频教程60:PHP与数据库代码开发规范 PHP100视频教程61:PHP+Mysql新闻系统的开发一 PHP100视频教程62:...
接下来我们一起来学习PHP中最常用的三种设计模式:单例设计模式、工厂设计模式和观察者设计模式。 单例设计模式 所谓单例模式,即在应用程序中最多只有该类的一个实例存在,一旦创建,就会一直存在于内存中! 单例...
本文实例讲述了PHP设计模式:桥连模式Bridge。分享给大家供大家参考,具体如下: 1. 概述 在软件系统中,某些类型由于自身的逻辑,它具有两个或多个维度的变化,那么如何应对这种“多维度的变化”?如何利用面向...
一、意图 将一个请求封装为一个对象,从而使用你可用不同的请求对客户进行参数化;对请求排队或记录请求日志,以及...二、命令模式结构图 三、命令模式中主要角色 命令(Command)角色:声明了一个给所有具体命令类
在不需要了解内部实现的前提下,遍历一个聚合对象的内部元素而又不暴露该对象的内部表示,这就是PHP迭代器模式的定义。 适用场景: 访问一个聚合对象的内容而无需暴露它的内部表示 支持对聚合对象的多种遍历 为遍历...
PHP中的设计模式 该存储库是项目的一部分。 它包含所有经典GoF设计模式PHP示例。 每个模式都包括两个示例: 概念示例显示了模式的内部结构,包括详细的注释。 RealWorld示例展示了如何在现实世界PHP应用程序中...