单例模式

1.单例模式

单例模式属于创建型模式,是创建对象的最佳方式
这种模式涉及到的一个单一的类,该类创建自己的对象,同时确保只有单个对象
这个类仅仅提供唯一对象访问方式,可以直接访问,不需要实例化对象

注意:	1.单例类只能有一个实例
		2.单例类必须自己给自己创建自己的唯一实例
		3.单例类必须给所有其他对象提供这一实例

1.单例模式主要思想:保证类中仅有一个实例,并且提供一个访问它的全局访问点
2.单例模式主要解决的问题:一个全局使用的类频繁的的创建和销毁
3.单例模式的使用场景:控制实例数目,节省系统资源的情况、生产唯一序列号、web计数器
4.关键思想:构造方法私有化
5.单例模式的优点:1)在内存中只有一个实例,减少内存开销,尤其是在频繁的创建和销毁实例的时候
2)避免对资源的多重占用
6.单例模式的缺点:1)没有接口
2)没有继承
3)与单一职责原则冲突
4)该类只关心内部逻辑,不关心外部如何实例化

1.1懒汉式

懒汉式:线程不安全
线程不安全严格意义上并不是单例模式,是用于理解单例模式的

package 懒汉式;

public class Singleton {
	private static Singleton instance;
	private Singleton(){
		System.out.println("Singleton的构造方法");
	}
	public static Singleton getinstance() {
		if(instance==null) {
			instance=new Singleton();
		}
		return instance;
	} 
	public static void main(String[] args) {
		// TODO Auto-generated method stub
		Singleton s=Singleton.getinstance();
		Singleton s2=Singleton.getinstance();
		System.out.println(s+"\n"+s2);
	}
}

输出结果:

Singleton的构造方法
懒汉式.Singleton@15db9742
懒汉式.Singleton@15db9742

这种情况对于普通调用似乎看不出什么问题,并且确实只创建了一个对象,调用了两次getinstance(),但是我们说过这个是基于线程不安全的

package 懒汉式;

class Singleton {
	private static Singleton instance;
	private Singleton(){
		System.out.println("Singleton的构造方法");
	}
	public static Singleton getinstance() {
		if(instance==null) {
			instance=new Singleton();
		}
		return instance;
	} 
}
package 懒汉式;

public class ThreadMain implements Runnable{

	public static void main(String[] args) {
		ThreadMain tm=new ThreadMain();
		ThreadMain tm1=new ThreadMain();
		Thread t=new Thread(tm1);
		Thread t1=new Thread(tm);
		t1.start();
		t.start();
	}

	@Override
	public void run() {
		Singleton s=Singleton.getinstance();
		System.out.println(s);
	}

}

输出结果:

Singleton的构造方法
Singleton的构造方法
懒汉式.Singleton@44ead434
懒汉式.Singleton@741102c

这个时候的运行结果就不再是创建一次对象了,这个时候一个线程进入Singleton类调用构造方法,另一个线程也进去调用构造方法,这样就产生了线程不安全,谁都可以进去访问,单例不单例有关系吗?所以这种情况的单例模式实际上只是用于理解单例的,很少使用

懒汉式:线程安全
线程安全是因为加了锁,保证单例
优点:第一次调用才初始化,避免造成浪费
缺点:必须加锁synchronized才能保证单例,但是加锁会影响效率

class Singleton1 {
	private static Singleton1 instance;
	private Singleton1(){
		System.out.println("Singleton的构造方法");
	}
	public static synchronized Singleton1 getinstance() {
		if(instance==null) {
			instance=new Singleton1();
		}
		return instance;
	} 
}
public class ThreadMain implements Runnable{
	public static void main(String[] args) {
		ThreadMain tm=new ThreadMain();
		ThreadMain tm1=new ThreadMain();
		Thread t=new Thread(tm1);
		Thread t1=new Thread(tm);
		t1.start();
		t.start();
	}
	@Override
	public void run() {
		Singleton1 s=Singleton1.getinstance();
		System.out.println(s);
	}

}

输出结果:

Singleton的构造方法
懒汉式.Singleton1@741102c
懒汉式.Singleton1@741102c

我们发现又正常了,因为我们加了synchronized实现了锁机制
懒汉式这个名称主要是由于这个模式是要用的时候就创建,不用就不创建,比较像懒汉一样,不管外面怎么乱糟糟,我只要是想用就用

1.2饿汉式

饿汉式是基于类加载机制实现的,避免了多线程的同步问题,不过,instance在类加载的时候就实例化了,大多数的单例都是调用getInstance方法
优点:没有加锁,执行效率高
缺点:类加载时就初始化,浪费内存,容易产生垃圾

package 饿汉式;

public class Singleton2 {
	private static Singleton2 instance=new Singleton2();
	private Singleton2(){
		System.out.println("Singleton2的构造方法");
	}
	public static Singleton2 getInstance() {
		return instance;
	}
}

package 饿汉式;

public class ThreadMain2 implements Runnable{

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		ThreadMain2 tm=new ThreadMain2();
		ThreadMain2 tm1=new ThreadMain2();
		Thread t=new Thread(tm);
		Thread t1=new Thread(tm1);
		t1.start();
		t.start();
	}

	@Override
	public void run() {
		Singleton2 s=Singleton2.getInstance();
		System.out.println(s);
	}
}

输出结果:

Singleton2的构造方法
饿汉式.Singleton2@741102c
饿汉式.Singleton2@741102c

因为它在类加载的时候就创建了对象,所以也叫饿汉式

1.3双检锁(双重校验锁)模式

DCL模式(Double-checked locking):采用双锁机制,线程安全,且双检锁机制可以大幅度降低synchronized带来的性能开销
多个线程在同一时间创建对象时,会通过加锁来保证只有一个线程来创建对象
对象创建好后,执行getInstance将不需要获取锁,直接返回已创建好的对象

package 双检锁模式;

public class Singleton3 {
	private volatile static Singleton3 instance;
	private Singleton3() {
		System.out.println("构造方法");
	}
	public static Singleton3 getInstance() {
		if(instance==null) {
			synchronized (Singleton3.class) {
				if(instance==null) {
					instance=new Singleton3();
				}
			}
		}
		return instance;
	}
}

package 双检锁模式;

public class ThreadMain3 {

	public static void main(String[] args) {
		// TODO Auto-generated method stub
		Singleton3 s3=Singleton3.getInstance();
		Singleton3 s4=Singleton3.getInstance();
		System.out.println(s3+"\n"+s4);
	}
}

1.4登记模式(静态内部类模式)重点

这种方式能达到双检锁方式相同的功效,但是实现要更简单,对静态域使用延迟初始化,应该使用这种方式而不是双检锁方式
这种方式同样利用classloader机制来保证初始化instance时只有一个线程

静态内部类模式和双检锁模式的区别:
1.双检锁只要Singleton类被加载了,那么instance就会被实例化,而静态内部类模式是Singleton类被加载了不一定被初始化,因为Singleton类没有被主动使用,只是通过显示调用getInstance方法时,才会显示加载Singleton类,从而实例化instance 
package 登记模式;

public class Singleton4 {
	private static class SingletonHolder{
		private static final Singleton4 INSTANCE=new Singleton4();
	}
	private Singleton4() {
		System.out.println("构造方法");
	}
	public static final Singleton4 getInstance() {
		return SingletonHolder.INSTANCE;
	}
}
package 登记模式;

public class ThreadMain4 {

	public static void main(String[] args) {
		Singleton4 s4=Singleton4.getInstance();
		Singleton4 s5=Singleton4.getInstance();
		System.out.println(s4+"\n"+s5);

	}
}

输出结果:

构造方法
登记模式.Singleton4@15db9742
登记模式.Singleton4@15db9742
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Carl·杰尼龟

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值