PHP中的设计模式:单例模式(译)
原文链接:http://coderoncode.com/2014/01/27/design-patterns-php-singletons.html
单例模式用于限制类实例化到单个对象,当整个系统只需要一个对象时这将有用。
单例模式被设计来确保只有一个单一的(因此得名单例)类实例,并且有一个全局的访问点。通过这个单一实例我们有一个全局的访问点并且可以延迟初始化。
一个基本的单例实现看起来像下面的例子:
<?php
/** Example taken from http://www.webgeekly.com/tutorials/php/how-to-create-a-singleton-class-in-php/ **/
class User
{
// Hold an instance of the class
private static $instance;
// The singleton method
public static function singleton()
{
if (!isset(self::$instance)) {
self::$instance = new __CLASS__;
}
return self::$instance;
}
}
$user1 = User::singleton();
$user2 = User::singleton();
$user3 = User::singleton();
?>
在上面的例子中3个变量都指向同一个对象。第一次调用会实例化出一个对象,而后续的调用都只返回该对象。
“在计算机编程中,延迟初始化是一种策略,直到第一次需要时才会去创建一个对象、计算一个值或者其他昂贵的过程。”
让我们看看显而易见而又被忽略的事实,并且讲讲为什么单例模式被许多开发者认为是一种“反模式”,尽管这么讲要高度依赖于所使用的框架和语言,然而PHP中的单例模式几乎被普遍认为是一种“反模式”。
“单例模式被许多人认为是一种反模式,反模式是一种解决方案,但是这种解决方案往往是无效的,甚至会带来高风险的反作用。”
单例模式是来自地狱的卵?
如果你之前读到过单例模式,你可能想知道到底我在表达什么?
单例模式是邪恶的,是一种“反模式”,并且绝不能使用它。光说单例模式是邪恶的还仅仅不够,我们需要知道为什么需要避免使用单例模式,这正是我想在这篇文章中阐明的。
单例模式被认为是一种“反模式”有几个原因,让我们来看看其中的一些原因:
单一职责模式
我们使用单一模式所面对的第一个问题便是:它的使用打破了单一职责原则。
单例对象有着用途和控制生产出的实例数量这两个职责,而单一职责原则则指出:每个类应该只有一个职责,并且这个职责应该由类来完全封装。
隐藏的依赖关系
什么是隐藏的依赖关系,以及这和单例模式是如何有关的?如果你看过我之前的有关依赖注入的文章,我们看到了如何把依赖关系通过参数传递给函数。
一个函数接收到的任何参数称为可见的依赖。另一方面,如果一个函数需要一个类似全局变量的东西(读取单例)来运作,那么这种依赖被认为是隐藏的。
这表明,如果不去看实际函数的实现的话,那么第三方是没有办法知道这个隐藏的依赖关系的。
“一个可见的依赖是一种开发者可以从类的接口看见到的依赖。如果一个依赖不能从类的接口看见的话,那么这个依赖就是一个隐藏的依赖”—jenkov.com
测试
伴随着这些问题,当我们对我们的应用程序进行单元测试是会有一个问题。合适的单元测试应该能像数据库那样独立地运行。单例模式让单元测试即使不是不可能的也会是困难的,因为单例模式有一个全局的状态。
这意味着,一旦实例化,那么这个实例便会仅仅围绕着这个单元测试。这可能会导致测试中意外地相互影响。
有个变通方法,那就是每次测试运行后好好地“清理”单例,然而我发现这仍然是生产上的禁忌并且很麻烦。
并不全是坏事,不是吗?
它们不能全是坏事,不是吗?好吧,让我们试着辩论下单例模式的好处和某些情况下它们是有用的:
1、调试日志:几乎所有的开发者都同意调试日志记录这种方式应该适用于每一个功能和每一部分的代码。单例模式可以在不伤害可读性、可测试性及可维护性上达到这个目的。
2、文件系统和数据库访问:访问文件系统和数据库用单例模式是由理由的,如果你需要一个文件系统或者数据库的全局访问点,单例模式在提供灵活性和可测试性上会小有帮助,这是有用的。
结论
单例模式、“反模式”和设计模式一般来讲并不是好的或者坏的。是什么让单例模式变成一个“反模式”并不是因为模式本身,而是因为单例模式往往很容易地被人们很差地实现。
如果被不正确地实现(这是经常发生的),任何模式都可能变成“反模式”。话虽这么说,就现代的PHP和现代的框架来说很难有赞成单例模式的情况。就我个人而言,对于单例模式,我看不出它有什么好处,恰恰看到了它有许多的缺点。
最后我明白的一点是,单例模式是一个有争议和偏见的话题。如果你有想补充任何在这篇文章中提到的观点,或者能提供有用的案例,我非常想看到你们在文章后面的评论。
更多文章请关注我的个人博客:http://www.nomoneynowife.com
PHP中的设计模式:单例模式(译)的更多相关文章
- [工作中的设计模式]单例模式singleton
一.模式解析: 单例模式是最简单和最常用的设计模式,面试的时候,不管新毕业的学生还是已经工作多年的筒子,对单例模式基本都能聊上两句.单例模式主要体现在如下方面: 1.类的构造函数私有化,保证外部不能直 ...
- 7.JAVA基础复习——JAVA中的设计模式单例模式
设计模式:是一套被反复使用.多数人知晓的.经过分类编目的.代码设计经验的总结.使用设计模式是为了可重用代码.让代码更容易被他人理解.保证代码可靠性. 总体来说设计模式分为23种三大类: 创建型模式,共 ...
- .net中的设计模式---单例模式
.net设计模式: Net设计模式实例之单例模式( Singleton Pattern) 一 : 单例模式的简介:(Brief Introduction) 单例模式(Singleton Patter ...
- .net中的设计模式---单例模式,涉及lock的用法
.客户端代码 static void Main(string[] args) { Singleton singleton2 = Singleton.GetInstance(); Singleton s ...
- Django中settings设计模式(单例模式)
配置文件: 需求:配置文件,默认配置和手动配置分开,参考django的配置文件方案,默认配置文件放在项目容器内部,只让用户做常用配置. /bin/settings.py(手动配置) PLUGIN_IT ...
- JavaScript 中常见设计模式整理
开发中,我们或多或少地接触了设计模式,但是很多时候不知道自己使用了哪种设计模式或者说该使用何种设计模式.本文意在梳理常见设计模式的特点,从而对它们有比较清晰的认知. JavaScript 中常见设计模 ...
- JS中常见设计模式总结
github: https://github.com/14glwu/FEInterviewBox/tree/master/JS%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F ...
- [Head First设计模式]抢票中的设计模式——代理模式
系列文章 [Head First设计模式]山西面馆中的设计模式——装饰者模式 [Head First设计模式]山西面馆中的设计模式——观察者模式 [Head First设计模式]山西面馆中的设计模式— ...
- [Head First设计模式]餐馆中的设计模式——命令模式
系列文章 [Head First设计模式]山西面馆中的设计模式——装饰者模式 [Head First设计模式]山西面馆中的设计模式——观察者模式 [Head First设计模式]山西面馆中的设计模式— ...
- Spring中的设计模式
[Spring中的设计模式] http://www.uml.org.cn/j2ee/201301074.asp [详解设计模式在Spring中的应用] [http://www.geek521.c ...
随机推荐
- Fiddler基本用法以及如何对手机抓包
一.Fiddler是什么? ·一种Web调试工具. ·可以记录所有客户端和服务器的http和https请求. ·允许监视.设置断点.修改输入输出数据. 官方文档(英文):http://docs.tel ...
- 哈希(3) java中的hashcode
看看jdk api中对object类的描述 1.http://docs.oracle.com/javase/7/docs/api/java/lang/Object.html 2首先你要知道每个对象都有 ...
- JavaScript高级程序设计22.pdf
操作节点 appendChild()用于向childNodes列表末尾添加一个节点,appendChild()返回新增的节点 var returnedNode=someNode.appendChild ...
- Linux常用命令收集
rsync -avH --progress felix/ /home/magnum/work-environment/ 同步本地文件夹,异常中断后仍然可以续传 rsync -avH --progres ...
- Yii PHP 框架分析(四)
作者:wdy http://hi.baidu.com/delphiss/blog/item/c15b314f05f9dfc0d0c86a26.html Yii应用的入口脚本最后一句启动了WebAppl ...
- Testing和Instrumentation(转)
Android提供了一系列强大的测试工具,它针对Android的环境,扩展了业内标准的JUnit测试框架.尽管你可以使用JUnit测试Android工程,但Android工具允许你为应用程序的各个方面 ...
- 修复南尼U盘
通常拿到手的U盘,用数码之家的ChipGenius一般检测不出主控型号,所以需要用到U盘量产工具来测. 量产工具注意版本 一般靠后的版本才能识别使用, . 如果识别不了==> 自动换成手动,U盘 ...
- maven项目文件乱码问题
今日碰到maven项目中的一个资源文件出现乱码,排查发现是在.setting文件夹下的org.eclipse.core.resources.prefs文件导致的. 修改前的编码规则如下: <sp ...
- (转)BAT及各大互联网公司2014前端笔试面试题--Html,Css篇
BAT及各大互联网公司2014前端笔试面试题--Html,Css篇 很多面试题是我自己面试BAT亲身经历碰到的.整理分享出来希望更多的前端er共同进步吧,不仅适用于求职者,对于巩固复习前端基础更是 ...
- Mac OS X 中安装JDK7
1. 下载 http://www.oracle.com/technetwork/java/javase/downloads/jdk7-downloads-1880260.html 当前版本1.7.0_ ...