为什么单例模式是邪恶的(译)

原文链接:http://blogs.msdn.com/b/scottdensmore/archive/2004/05/25/140827.aspx

虽然这篇文章不是我写的,但我完全同意文章中的观点。Brian Button可能是我所知道的最有才的人之一。我相信他会喜欢你们的反馈的。

1、单例模式经常被用来为某些服务提供一个全局访问点
没错,是可以这么做,但代价是什么呢?众所周知,单例模式为你的应用程序中的某些服务提供全局访问点,这样你就不必到处传递一个该服务的引用。这和一个全局变量有什么区别呢?(记住,全局变量是不好的,不是吗?)最终会发生的事情便是你设计中的依赖关系是隐藏在代码中的,这种依赖关系不能够通过检查你的类和方法的接口所见。你必须检查代码以便准确地知道你的类中都用了哪些其他对象。这可能会是不清楚的(译者注:不知道是不是这么翻译,原文:This is less clear than it could be)。冲动地创建一个全局性的东西来避免把其传来传去是你设计中的异味,这不是全局变量或者单例的功能。如果你更仔细地检查你的设计,你几乎总是能想到一种不必把“流浪数据”四处传递到每个对象和方法的更好的设计。

2、单例模式允许你限制你所创建的对象的数量
这也是对的,但是现在你混合了两个不同的职责到同一个类中了,这是违反了单一职责原则的。一个类不应该关心它是否是一个单例,它应该仅仅关心它的业务职责。如果你想限制某些类的实例化能力,创建一个封装了创建并且如你所愿的那样限制创建能力的工厂或者对象生成器,这样创建职责便从业务实体职责中划分出来了。

3、单例模式促进了类之间的紧耦合
使代码是可测试的一个基本属性便是它和其周围环境是松耦合的。这个属性允许你在测试过程中为了实现特定的测试目标(想想模拟对象)替换备用的合作者。单例模式把单例对象的确切类型紧耦合在一起,丧失了使用多态来替换的机会。一个更好的选择,如上面第一点所讨论到的,便是改变你的设计以便允许你传递对象的引用到你的类和方法中,这样可以减轻上述的耦合问题。

4、单例模式在程序的持续过程中一直保存着上一次的状态
持久状态是单元测试的敌人。让单元测试有效的事情之一便是每个测试必须独立于其他所有测试。如果不是这样,那么测试运行的顺序会影响到测试的结果。这可能会导致测试在不应该失败的地方失败,甚至更坏的事情是仅仅因为测试的运行顺序导致测试通过。这可能会隐藏住bugs并且是邪恶的。防止状态从一个测试携带到另一个测试的一个很好办法便是避免使用静态变量。单例模式,就其本质而言,依赖于在一个静态变量中保存着一个实例。这是测试依赖的一个挑战,可以通过传递对象的引用到你的类和方法中来避免。

希望这篇文章或多或少地阐释了我对于单例模式观点。我有一个从google或者其他地方找到的一个小链接集,包括Jim Hyslop和Herb Sutter,他们同样分享了这些观点。如果你喜欢他们那么让我知道吧。

转载于:https://www.cnblogs.com/shower/p/3738754.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/359859.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

java properties 保存_Java 读写Properties配置文件

转自:https://www.cnblogs.com/xudong-bupt/p/3758136.html1.Properties类与Properties配置文件Properties类继承自Hashtable类并且实现了Map接口,也是使用一种键值对的形式来保存属性集。不过Properties有特殊的地方,就是它的键和值都是字符…

自动加密可序列化的类

在Coursera安全性最高项目的验尸讨论中提出了一个疯狂的想法。 类可以在序列化期间对其自身进行加密吗? 这主要是一项学术性的“假设”练习。 很难想到这样一种情况,我们希望在持久性期间依靠对象自加密而不是使用显式加密机制。 我只能确定一种情况&am…

java垃圾回收机制优化_JVM性能优化--Java的垃圾回收机制

一、Java内存结构1、Java堆(Java Heap)java堆是java虚拟机所管理的内存中最大的一块,是被所有线程共享的一块内存区域,在虚拟机启动时创建。此内存区域的唯一目的就是存放对象实例,这一点在Java虚拟机规范中的描述是:所有的对象实…

《linux 网卡别名的添加和绑定》RHEL6

网卡别名的配置: 这个和ifconfig临时修改网卡ip 差不多,但是不一样。都是临时的,只要重启电脑就没了。 配永久的ip别名: cp ifcfg-eth0 ifcfg-eth0:0 vim ifcfg-eth0:0 这样做也能出来,对不对就不知道了 重启网络就ok…

NGUI中UILabel使用url标签的一个bug

在NGUI里,UILabel控件可以支持一些简单功能的标签,使文本显示更丰富及实现类似超链接的功能。但是在使用的时候发现了NGUI3.5.9版本里存在着一个bug。不过还好修复这个bug也很简单。 在UILabel中支持[urllink]text[/url]的方式来定义类超链接的文本。bug…

web前端模块化开发_真正的模块化Web应用程序:为什么没有开发标准?

web前端模块化开发OSGI , SpringSource , Jboss模块 ,J2EE和清单永远不会结束。所有这些技术都向他们的最终用户/开发人员保证了相同的东西,或多或少是Java模块化Web应用程序(?)。 但是&#xf…

[转]Oracle DB管理内存

• 描述SGA 中的内存组件• 实施自动内存管理• 手动配置SGA 参数• 配置自动PGA 内存管理内存管理:概览DBA 必须将内存管理视为其工作中至关重要的部分,因为:• 可用内存空间量有限• 为某些类型的功能分配更多内存可提高整体性能• 自动优化…

java request获取文件_request获取路径方式

从request获取各种路径总结request.getRealPath("url"); // 虚拟目录映射为实际目录request.getRealPath("./"); // 网页所在的目录request.getRealPath("../"); // 网页所在目录的上一层目录request.getContextPath(); // 应用的web目录的…

servlet 3.0异步_Servlet 3.0异步处理可将服务器吞吐量提高十倍

servlet 3.0异步Servlet是Java中处理服务器端逻辑的主要组件,新的3.0规范引入了一些非常有趣的功能,其中异步处理是最重要的功能之一。 可以利用异步处理来开发高度可伸缩的Web应用程序。 使用此功能可以有效地构建Web 2.0站点和AJAX应用程序。 我们的JC…

Android自定义进度条-带文本(文字进度)的水平进度条(ProgressBar)

/** * 带文本提示的进度条 */ public class TextProgressBar extends ProgressBar { private String text; private Paint mPaint; public TextProgressBar(Context context) { super(context); initText(); } public TextProgressBar(Context context, AttributeSet attrs, in…

java 获取服务器网络名_java-siger java使用siger 获取服务器硬件信息(CPU 内存 网络 io等) - 下载 - 搜珍网...

java读取系统信息/java读取系统信息/hyperic-sigar-1.6.4/java读取系统信息/hyperic-sigar-1.6.4.zipjava读取系统信息/hyperic-sigar-1.6.4/AUTHORSjava读取系统信息/hyperic-sigar-1.6.4/bindings/java读取系统信息/hyperic-sigar-1.6.4/bindings/dotnet/java读取系统信息/hy…

《无码的青春》第四章 程序员的二象性,左手流氓,右手疯子

“道哥,你都30了,怎么还不考虑成家的事情啊”,张小凡问到。 其实他不了解道哥的过去,当年道哥搞javaABC论坛的时候,有一个javaABC官方扯淡群,里面聚集了一群程序员,从不讨论技术,唯…

java导出highcharts_Highcharts导出代码Java版

Highcharts是一个用纯JavaScript编写的图表库,提供了一个交互式的图表添加到您的网站或Web应用程序的简单方法。Highcharts目前支持线,样条,面积,areaspline,柱形图,条形图,饼图和散点图类型。同…

smartgwt_SmartGWT入门,提供出色的GWT界面

smartgwtSmartGWT简介 我最近开始使用SmartGWT ,它是一个基于GWT的框架,该框架为您的应用程序UI提供了一个全面的小部件库,并为服务器端的数据管理提供了帮助。 您可以在SmartGWT展示柜上查看其漂亮的功能。 我准备了一个简短的“入门”指南…

java录入会员信息_java-第三章-升级我行我素购物管理系统,实现会员信息录入的功能...

import java.util.Scanner;public class A01 {/*** param args*/public static void main(String[] args) {// TODO Auto-generated method stubScanner input new Scanner (System.in);System.out.println("我行我素购物管理>客户信息管理>添加客户信息");Sy…

Beaglebone Back学习五(PWM测试)

PWM测试 参考链接 1 Enable PWM on BeagleBone with Device Tree overlays 2 Using PWM on the Beaglebone Black 3 Beaglebone Coding 101: Buttons and PWM 4 Using PWM outputs 5 beaglebone-black-cpp-PWM 6 Enabling PWM Support in the kernel 7转载于:https://www.cnblo…

CUBA平台的理念

最近发生了很多事。 在CUBA于6月1日正式发布之后,我们推出了一个新版本,在一些Java网站上发布了我们的第一篇文章,并在伦敦的Devoxx UK会议上介绍了该平台 。 但是在热潮继续之前,大约是时候阐明CUBA背后的哲学了。 与企业软件开…

mysql orderby多个_MySQL OrderBy

MySQL会为每个线程分配一个内存(sort_buffer)用于排序,该内存小大为 sort_buffer_size如果排序的数量小于 sort_buffer_size,排序将会在内存中完成。如果排序数据量很大,内存中无法存下这么多数据,则会使用磁盘临时文件来辅助排序…

java 双重检查锁_Java中可怕的双重检查锁定习惯用法

java 双重检查锁本文讨论的问题不是新问题,但即使是经验丰富的开发人员也仍然很棘手。 单例模式是常见的编程习惯用法。 但是,当与多个线程一起使用时,必须进行某种类型的同步,以免破坏代码。 在相关文章中,我们的JCG合…

mysql-bin.index找不到_MySQL不能启动 mysql-bin.index' not found (Errcode: 13)

配置复制,添加如下内容到/etc/my.cnf:log-bin/var/lib/mysql/binlogs/mysql-binmax_binlog_size100Mexpire_logs_days5sync_binlog1binlog_cache_size1Mbinlog-formatROW结果重启的时候,报错:-[root www.linuxidc.com mysql]# /et…