首先写个单例,懒汉模式:

public class SingleDemo {
private static SingleDemo s = null;
private SingleDemo(){}
public static SingleDemo getInstance(){
if(s == null){
s = new SingleDemo();
}
return s;
}
}

写个测试类:

public class ThreadDemo3 {

    public static void main(String[] args) {
SingleDemo s1 = SingleDemo.getInstance();
SingleDemo s2 = SingleDemo.getInstance();
System.out.println(s2 == s2);
}
}

运行结果一直都是true,说明单线程下是没问题的,下面写个多线程来访问单例

public class ThreadTest implements Runnable {
//存放单例对象,使用Set是为了不存放重复元素
public Set<SingleDemo> singles = new HashSet<SingleDemo>();
@Override
public void run() {
//获取单例
SingleDemo s = SingleDemo.getInstance();
//添加单例
singles.add(s);
}
}

使用多线程并发访问单例:

public class ThreadDemo3 {

    public static void main(String[] args) {
// SingleDemo s1 = SingleDemo.getInstance();
// SingleDemo s2 = SingleDemo.getInstance();
// System.out.println(s2 == s2);
ThreadTest t = new ThreadTest();
new Thread(t).start();
new Thread(t).start();
new Thread(t).start();
new Thread(t).start();
new Thread(t).start();
new Thread(t).start();
new Thread(t).start();
new Thread(t).start();
System.out.println(t.singles);
}
}

运行结果如下:

[com.persagy.thread.SingleDemo@1bc4459, com.persagy.thread.SingleDemo@150bd4d]

[com.persagy.thread.SingleDemo@12b6651]

说明有线程并发访问安全问题,获取的不一定都是同一个实例

如何解决线程安全问题呢?

当然使用同步锁机制了啊

下面改进单例:

public class SingleDemo {
private static SingleDemo s = null;
private SingleDemo(){}
public static synchronized SingleDemo getInstance(){
if(s == null){
s = new SingleDemo();
}
return s;
}
}

加入同步函数后线程安全问题解决了

运行多次都是获取同一个实例,不会出现2个实例的情况了

[com.persagy.thread.SingleDemo@12b6651]

但是在多线程并发访问的情况下,每个线程每次获取实例都要判断下锁,效率比较低,为了提高效率,我加入了双重判断的方法,解决了效率的问题

代码如下;

public class SingleDemo {
private static SingleDemo s = null;
private SingleDemo(){}
public static SingleDemo getInstance(){
/*如果第一个线程获取到了单例的实例对象,
* 后面的线程再获取实例的时候不需要进入同步代码块中了*/
if(s == null){
//同步代码块用的锁是单例的字节码文件对象,且只能用这个锁
synchronized(SingleDemo.class){
if(s == null){
s = new SingleDemo();
}
}
}
return s;
}
}

用这种方式解决了懒汉式的线程安全问题,也提高了效率,但是在实际开发中还是用饿汉式的比较多,毕竟这个代码比较多,比较繁琐。

最新文章

  1. 用“MEAN”技术栈开发web应用(三)用mongodb搭建数据库
  2. .net中xml文件的导入使用(包括创建xml和导入xml)
  3. VS2015 自动添加头部注释
  4. [Android]android studio预览视图时报错
  5. [RabbitMQ] Connection failed
  6. JAVA 23种设计模式(转)
  7. 标准SAP中的物料类型
  8. Android 线程通讯类Handler
  9. 完全掌握Android Data Binding
  10. BZOJ_1180_[CROATIAN2009]_OTOCI_(LCT)
  11. 单位冲击响应与频响以及FIR实现代码(C语言)(转)
  12. yeoman 使用问题总结
  13. chrome console的使用 : 异常和错误的处理 – Break易站
  14. CSS文本属性 文本阴影text-shadow 换行 text-overflow
  15. vsftp配置文件详解
  16. Django之用户认证组件
  17. 20135327--linux内核分析 实践二
  18. git获取一个版本相对于另一个版本新增,修改,删除的文件
  19. PAT1135(红黑书的判定)
  20. 动态生成PictureBox控件,涉及:PictureBox控件和flowLayoutPanel面板

热门文章

  1. 精读《12 个评估 JS 库你需要关心的事》
  2. JavaScript - 库 jQuery
  3. java中的构造方法(2013-05-05-bd 写的日志迁移
  4. python-9-IO编程
  5. 基于itchat定制聊天机器人
  6. div+css实现双飞翼布局
  7. Android通过用代码画虚线椭圆边框背景来学习一下shape的用法
  8. 《Cracking the Coding Interview》——第12章:测试——题目5&#183;
  9. android什么时候会产生ANR
  10. 最近做group assignment需要些加密的知識