单例模式在多线程环境下的lazy模式为什么要加两个if(instance==null)
刚才在看阿寻的博客”C#设计模式学习笔记-单例模式“时,发现了评论里有几个人在问单例模式在多线程环境下为什么lazy模式要加两个if进行判断,评论中的一个哥们剑过不留痕,给他们写了一个demo来告诉他们为什么。
我看了一下这个demo,确实说明了这个问题,但我认为不够直观,呵呵,于是我就稍微的改了一下。
这是剑过不留痕的demo
using System;
using System.Threading; namespace SingletonPattern
{
class Program
{
static Thread[] threads; static void Main(string[] args)
{
int threadsCapacity; Console.WriteLine("请输入开启的线程数:");
string s;
do
{
s = Console.ReadLine();
if (!int.TryParse(s, out threadsCapacity))
{
Console.WriteLine("数字格式不正确,请输入开启的线程数:");
continue;
}
else if ( > threadsCapacity)
{
Console.WriteLine("数字应不小于 1,请输入开启的线程数:");
continue;
}
else
{
break;
}
} while (true); threads = new Thread[threadsCapacity];
Program pg = new Program();
for (int i = ; i < threads.Length; i++)
{
Thread thread = new Thread(new ThreadStart(Singleton.GetIns));
thread.Name = "线程 " + i;
threads[i] = thread;
} for (int i = ; i < threads.Length; i++)
{
threads[i].Start();
} Console.ReadKey();
}
} class Singleton
{
private static Singleton instance;
private static object _lock = new object();
private static int instanceQuantity = ; private Singleton()
{ } public static Singleton GetInstance()
{
if (instance == null)
{
// 挂起线程,确保所有的线程都执行到这并等待
Thread.Sleep(); lock (_lock)
{
//if (instance == null)
//{
// instance = new Singleton();
// ++instanceQuantity;
// Console.WriteLine(Thread.CurrentThread.Name + " 生成第 " + instanceQuantity + " 实例!");
//} instance = new Singleton();
++instanceQuantity;
Console.WriteLine(Thread.CurrentThread.Name + " 生成第 " + instanceQuantity + " 实例!");
}
}
return instance;
} public static void GetIns()
{
GetInstance();
}
}
}
下面是我修改过得demo,修改的地方主要在GetInstance()这个方法内部,其他地方没动,所以我只贴出修改代码的地方,免得占空间!
public static Singleton GetInstance()
{
if (instance == null)
{
// 挂起线程,确保所有的线程都执行到这并等待
Console.WriteLine("我是" + Thread.CurrentThread.Name + " ,instance等于null,哦,对象还没创建呢,看来我有机会哦!");
Thread.Sleep(); lock (_lock)
{
Console.WriteLine("我是" + Thread.CurrentThread.Name + " ,instance等于null,哈哈,我锁,我是第一个到的!创建对象的任务就交给我了!");
if (instance == null)
{
instance = new Singleton();
++instanceQuantity;
Console.WriteLine("我是" + Thread.CurrentThread.Name + " ,哼哼,对象是我创建的!");
}
else
{
Console.WriteLine("我是"+Thread.CurrentThread.Name + ",虽然我之前判断instance等于null,但现在判断的时候,却不等null了,唉,还是被别人快了一步!不过好在我判断了,要不就多创建了一个,失去了单例模式的意义!");
}
//Console.WriteLine(Thread.CurrentThread.Name + " 生成第 " + instanceQuantity + " 实例!");
//instance = new Singleton();
//++instanceQuantity;
//Console.WriteLine(Thread.CurrentThread.Name + " 生成第 " + instanceQuantity + " 实例!");
}
}
return instance;
}
下面是运行的结果,多运行几遍就可能出现这个结果了。

文字描述:
虽然线程0先进的第一个if,但创建对象的确实线程1,而此时,线程0已经判断了instance==null,所以他还会再创建一个对象,因为并没有人告诉线程0,线程1已经创建过对象了,而内部的这个if,就是为了告诉别人,对象我已经创建过了!其他人就不要再创建了,不知道我这样解释,清楚不!
单例模式在多线程环境下的lazy模式为什么要加两个if(instance==null)的更多相关文章
- 一个由单例模式在多线程环境下引发的 bug
问题症状 HTTP 日志系统,老是出现日志信息覆盖的情况.比如同时调用 A 接口和 B 接口,B 接口请求响应信息变成了 A 接口请求响应相关信息.这个问题在并发量大的情况下越来越严重. 问题初步分析 ...
- Java指令重排序在多线程环境下的应对策略
一.序言 指令重排在单线程环境下有利于提高程序的执行效率,不会对程序产生负面影响:在多线程环境下,指令重排会给程序带来意想不到的错误. 本文对多线程指令重排问题进行复原,并针对指令重排给出相应的解决方 ...
- SQLite在多线程环境下的应用
文一 SQLite的FAQ里面已经专门说明,先贴出来.供以后像我目前的入门者学习. (7) 多个应用程序或者同一个应用程序的多个例程能同时存取同一个数据库文件吗? 多进程可以同时打开同一个数据库,也可 ...
- 多线程环境下非安全Dictionary引起的“已添加了具有相同键的项”问题
问题: 代码是在多线程环境下,做了简单的Key是否存的判断, 测试代码如下: public class Program { static Dictionary<string, Logger> ...
- C#多线程环境下调用 HttpWebRequest 并发连接限制
C#多线程环境下调用 HttpWebRequest 并发连接限制 .net 的 HttpWebRequest 或者 WebClient 在多线程情况下存在并发连接限制,这个限制在桌面操作系统如 win ...
- C++多线程环境下的构造函数
多线程的环境里,我们总不可避免要使用锁.于是一个常见的场景就是: class ObjectWithLock { private: std::mutex mtx_; SomeResType shared ...
- Asp.Net在多线程环境下的状态存储问题
在应用开发中,我们经常需要设置一些上下文(Context)信息,这些上下文信息一般基于当前的会话(Session),比如当前登录用户的个人信息:或者基于当前方法调用栈,比如在同一个调用中涉及的多个层次 ...
- HttpClient在多线程环境下踩坑总结
问题现场 在多线程环境下使用HttpClient组件对某个HTTP服务发起请求,运行一段时间之后发现客户端主机CPU利用率呈现出下降趋势,而不是一个稳定的状态. 而且,从程序日志中判断有线程处于han ...
- 多线程环境下的UI异步操作
转自原文 多线程环境下的UI异步操作 解决VS中,线程间不可互操作的问题,一揽子解决方案: 一.首先,定义一个类:SetControlProperty using System.Reflection; ...
随机推荐
- 2017北京ICPC C题 Graph
#1629 : Graph 时间限制:4000ms 单点时限:4000ms 内存限制:256MB 描述 The country contains N cities numbered from 1 to ...
- noip模拟赛 读
分析:感觉很像是贪心,但是直接贪找不到方法.一个暴力的想法是枚举最小步数,然后看每个指针能够覆盖到的位置,看看能不能覆盖到所有点.这个求最大覆盖就有点贪心的思想,因为给的ai,bi都是递增顺序的,考虑 ...
- Mysql 练习题 及 答案
--1.学生表 Student(S,Sname,Sage,Ssex) --S 学生编号,Sname 学生姓名,Sage 出生年月,Ssex 学生性别 --2.课程表 Course(C,Cname,T ...
- codevs1004 四子连棋
题目描述 Description 在一个4*4的棋盘上摆放了14颗棋子,其中有7颗白色棋子,7颗黑色棋子,有两个空白地带,任何一颗黑白棋子都可以向上下左右四个方向移动到相邻的空格,这叫行棋一步,黑白双 ...
- 玲珑杯 ACM Round #10
A 题意:给长度为n的序列染黑白色,要求连续的黑的格子数量<=a,连续的白的格子数量<=b,问方案总数,有多个询问 分析:递推 注意数据范围,是可以O(n)做的,所以可以直接递推 B 题意 ...
- vsftpd conf 解釋
Linux中vsFTP位置约定:/usr/sbin/vsftpd ---- VSFTPD的主程序/etc/rc.d/init.d/vsftpd ---- 启动脚本/etc/vsftpd/vsftpd. ...
- C++成员对齐方式探讨
本文參考了<高质量程序设计指南--C++/C语言>一书 有不妥之处恳请指正 一.自然对齐 某些基于RISC(精简指令集)的CPU比方SPARC.PowerPC等.採用高字节和高字在低地址存 ...
- 关于PROFIBUS Master(H)不能正确识别并处理 DP-Slave 回复的RS帧的一些思考
图1.是在測试过程中,发现PROFIBUS Master(H)不能正确识别并处理 DP-Slave 回复的RS帧.引起Slave回复 RS 帧的操作是"断开Slave与Master之间的PR ...
- 继承自TWinControl的控件不能在设计期间接受子控件,用代码设置子控件却可以(它的自绘是直接改写PaintWindow虚函数,而不是覆盖Paint函数——对TWinControl.WMPaint又有新解了)
这个控件直接继承自TWinControl,因此不是改写Paint;函数,而是直接改写PaintWindow虚函数,它在VCL框架里被直接调用,直接就把自己画好了(不用走给控件Perform(WM_Pa ...
- Fiddler抓取https请求,解决“证书错误”警告
要抓取走HTTPS内容,Fiddler必须解密HTTPS流量. 但是,浏览器将会检查数字证书,并发现会话遭到窃听.为了骗过浏览 器,Fiddler通过使用另一个数字证书重新加密HTTPS流量. Fid ...