原文链接: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中的设计模式:单例模式(译)的更多相关文章

  1. [工作中的设计模式]单例模式singleton

    一.模式解析: 单例模式是最简单和最常用的设计模式,面试的时候,不管新毕业的学生还是已经工作多年的筒子,对单例模式基本都能聊上两句.单例模式主要体现在如下方面: 1.类的构造函数私有化,保证外部不能直 ...

  2. 7.JAVA基础复习——JAVA中的设计模式单例模式

    设计模式:是一套被反复使用.多数人知晓的.经过分类编目的.代码设计经验的总结.使用设计模式是为了可重用代码.让代码更容易被他人理解.保证代码可靠性. 总体来说设计模式分为23种三大类: 创建型模式,共 ...

  3. .net中的设计模式---单例模式

    .net设计模式: Net设计模式实例之单例模式( Singleton Pattern)  一 : 单例模式的简介:(Brief Introduction) 单例模式(Singleton Patter ...

  4. .net中的设计模式---单例模式,涉及lock的用法

    .客户端代码 static void Main(string[] args) { Singleton singleton2 = Singleton.GetInstance(); Singleton s ...

  5. Django中settings设计模式(单例模式)

    配置文件: 需求:配置文件,默认配置和手动配置分开,参考django的配置文件方案,默认配置文件放在项目容器内部,只让用户做常用配置. /bin/settings.py(手动配置) PLUGIN_IT ...

  6. JavaScript 中常见设计模式整理

    开发中,我们或多或少地接触了设计模式,但是很多时候不知道自己使用了哪种设计模式或者说该使用何种设计模式.本文意在梳理常见设计模式的特点,从而对它们有比较清晰的认知. JavaScript 中常见设计模 ...

  7. JS中常见设计模式总结

    github: https://github.com/14glwu/FEInterviewBox/tree/master/JS%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F ...

  8. [Head First设计模式]抢票中的设计模式——代理模式

    系列文章 [Head First设计模式]山西面馆中的设计模式——装饰者模式 [Head First设计模式]山西面馆中的设计模式——观察者模式 [Head First设计模式]山西面馆中的设计模式— ...

  9. [Head First设计模式]餐馆中的设计模式——命令模式

    系列文章 [Head First设计模式]山西面馆中的设计模式——装饰者模式 [Head First设计模式]山西面馆中的设计模式——观察者模式 [Head First设计模式]山西面馆中的设计模式— ...

  10. Spring中的设计模式

    [Spring中的设计模式] http://www.uml.org.cn/j2ee/201301074.asp [详解设计模式在Spring中的应用]    [http://www.geek521.c ...

随机推荐

  1. 解决eclipse插件svn不显示svn信息和显示的信息为数字的问题

    1.选择window-->preferences如下图 通过上面步骤svn信息便显示了 2.解决显示的信息为数字问题 选择svn-label decoration format里面的author ...

  2. HTML5 Canvas核心技术—图形、动画与游戏开发.pdf3

    路径与子路径 在某一时刻,canvas之中只能有一条路径存在,Canvas规范称之为“当前路径”(current path),这条路径却可以包含很多子路径(subpath),子路径是由两个或者更多点组 ...

  3. sublime中文乱码解决办法

    欲解决问题,关键在于让Sublime Text 2支持GB2312和GBK.步骤如下: 1.安装Sublime Package Control. 在Sublime Text 2上用Ctrl+-打开控制 ...

  4. Educational Codeforces Round 3

    A. USB Flash Drives 水题,排序即可 ]; int main() { int n,m; scanf("%d%d",&n,&m); ;i<n; ...

  5. CMake 入门实战 | HaHack

    CMake 入门实战 | HaHack undefined

  6. date 、cal、bc

    date cal date是显示日期与时间 date +%Y/%M/%D date +%Y/%m/%d date +%H:%M cal cal 2016

  7. [html][css]让文字在div中居中的方法[转]

    转至:http://dreamweaver.abang.com/od/divcss/a/vertical-align.htm 一.行高(line-height)法 如果要垂直居中的只有一行或几个文字, ...

  8. C++:复制构造函数在什么时候被调用?

    这个问题不是疑问了,查了一下国外网站,总结一下.假设Person是一个类,复制构造函数的调用会在以下几种情况下发生: 1.对象在创建时使用其他的对象初始化 Person p(q); //此时复制构造函 ...

  9. inline-block元素之间出现间隙

    一.问题 这里部分的组成是一个input框和一个a按钮,然后a标签为了设置它的width和height我让他display:inline-block(行内元素以块级元素显示内容).神奇的一幕出现了,两 ...

  10. 【转】Android 混淆代码总结

    http://blog.csdn.net/lovexjyong/article/details/24652085 为了防止自己的劳动成果被别人窃取,混淆代码能有效防止被反编译,下面来总结以下混淆代码的 ...