原文链接: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. MFC如何获取编辑框中输入的内容

    1.GetDlgItemText() 2.先用 GetDlgItem(编辑框的ID)获取指向编辑框的指针.再用GetWindowText函数将获取内容保存至指定的字符数组里. 3.使用 GetDlgI ...

  2. 浙大PTA - - 堆中的路径

    题目链接:https://pta.patest.cn/pta/test/1342/exam/4/question/21731 本题即考察最小堆的基本操作: #include "iostrea ...

  3. ios 实现跳转到评价界面的两种方式

    要想在App内跳转到特定App的详情页或者评论页,首先需要获取到App的id.在 iTunes Connect网站上登陆之后,选择“我的App”,然后点击某个特定的App进入,在App信息的综合信息中 ...

  4. 基于Unity3D的AOP使用思路

    考虑到AOP的特性,并参考Python的装饰模式(AOP类似),在做开发时,可以使用AOP的思想做一些工作上的简化(比如:运行时的Range属性.内存/存档加密属性等等). 但是AOP(使用的Post ...

  5. 理解asp.net中DropDownList编辑数据源,绑定数据库数据。

    一.理解asp.net绑定数据库 终于学习到了连接数据库部分的内容,杨中科老师视频看起来挺轻松的,如果是高清版就更ok了. 我发现我学习新的编程语言会有一个特点,都想要赶紧学习数据库,数据就是一切,有 ...

  6. cocos2d-x 事件分发机制 ——加速计事件监听

    加速计事件监听机制 在上一篇中介绍了cocos2d-x中的触摸事件机制,这篇来介绍下游戏中也常常常使用到的加速计事件,这些都是游戏中的常常要用到的. 移动设备上一个非常重要的输入源是设备的方向.大多数 ...

  7. [D3] 3. Scaling Basics

    d3.scale.linear() <!DOCTYPE html> <html> <head lang="en"> <meta chars ...

  8. android 76 使用get post方式提交数据

    get方式: package com.itheima.getmethod; import java.io.InputStream; import java.net.HttpURLConnection; ...

  9. NVMe 图解

    http://www.ssdfans.com/?p=1143#rd&sukey=3997c0719f151520989740bb972a716fdb2dbab623808d16acd5075b ...

  10. JavaRTS-DTraceProvider

    https://docs.oracle.com/javase/realtime/doc_2.0_u1/release/JavaRTSDTraceProvider.html