单例模式(Singleton Pattern)是 Java 中最简单的设计模式之一。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。
这种模式涉及到一个单一的类,该类负责创建自己的对象,同时确保只有单个对象被创建。这个类提供了一种访问其唯一的对象的方式,可以直接访问,不需要实例化该类的对象。
单例模式有分为饿汉式和懒汉式
特点:
应用实例:
优点:
缺点:没有接口,不能继承,与单一职责原则冲突,一个类应该只关心内部逻辑,而不关心外面怎么样来实例化。
使用场景:
注意事项:getInstance() 方法中需要使用同步锁 synchronized (Singleton.class) 防止多线程同时进入造成 instance 被多次实例化。
代码
//单例模式-饿汉式
public class SingletonDemo {
public static void main(String[] args) {
//编译错误:无法实例化
// Singleton singleton = new Singleton();
//正确获取对象的方法
Singleton singleton = Singleton.getINSTANCE();
singleton.hello();
}
}
class Singleton{
//创建一个本身对象
private static final Singleton INSTANCE = new Singleton();
//让构造方法为private,这样该类就不会被实例化
private Singleton(){}
//创建一个获取对象的方法
public static Singleton getINSTANCE() {
return INSTANCE;
}
public void hello(){
System.out.println("Hello World! ——单例模式-饿汉式");
}
} 结果
Hello World! ——单例模式-饿汉式
这种方式是最基本的实现方式,这种实现最大的问题就是不支持多线程。因为没有加锁 synchronized,所以严格意义上它并不算单例模式。
这种方式 lazy loading 很明显,不要求线程安全,在多线程不能正常工作。
代码
//单例模式-懒汉式
public class SingletonDemo2 {
public static void main(String[] args) {
Singleton2 singleton = Singleton2.getInstance();
singleton.hello();
}
}
class Singleton2{
private static Singleton2 instance;
private Singleton2(){}
public static Singleton2 getInstance() {
if (instance == null){
instance = new Singleton2();
}
return instance;
}
public void hello(){
System.out.println("Hello World! ——单例模式-懒汉式");
}
}
结果
Hello World! ——单例模式-懒汉式
描述:这种方式具备很好的 lazy loading,能够在多线程中很好的工作,但是,效率很低,99% 情况下不需要同步。
优点:第一次调用才初始化,避免内存浪费。
缺点:必须加锁 synchronized 才能保证单例,但加锁会影响效率。
getInstance() 的性能对应用程序不是很关键(该方法使用不太频繁)。
代码
//单例模式-懒汉式
public class SingletonDemo3 {
public static void main(String[] args) {
Singleton3 singleton = Singleton3.getInstance();
singleton.hello();
}
}
class Singleton3{
private static Singleton3 instance;
private Singleton3(){}
public synchronized static Singleton3 getInstance() {
if (instance == null){
instance = new Singleton3();
}
return instance;
}
public void hello(){
System.out.println("Hello World! ——单例模式-懒汉式");
}
}
结果
Hello World! ——单例模式-懒汉式
public class Singleton{
private volatile static Singleton instance;
//私有构造函数
private Singleton{
}
//唯一实例
public static Singleton getInstacne(){
if(instance == null){
synchronized(Singleton.class){
if(instance == null){
instance = new Singleton();
}
}
}
return instance;
}
} // 饿汉式,利用同一时间只能有一个线程对类进行初始化来保证线程安全
public class Clazz {
private Clazz() {}
private static Clazz instance = new Clazz();
public static Clazz getInstance() {
return instance;
}
}
// 懒汉式1,内部类方式。同样利用同一时间只能有一个线程对类进行初始化来保证线程安全
public class Clazz {
private Clazz() {}
private static class InnerClazz() {
public static Clazz instance = new Clazz();
}
public static Clazz getInstance() {
return InnerClazz.instance;
}
}
// 懒汉式2,双重检测方式。可能会有多个线程同时进入第一个判断,之后多个线程依次进入同步代码块,第一个线程完成初始化,其他线程直接退出
//
public class Clazz {
private Clazz() {}
private static Clazz instance;
public static Clazz getInstance() {
if(instance == null) {
synchronized (Clazz.class) {
if(instance == null) {
instance = new Clazz();
}
}
}
return instance;
}
} 单例模式(Singleton Pattern)是 Java 中最简单的设计模式之一。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。
这种模式涉及到一个单一的类,该类负责创建自己的对象,同时确保只有单个对象被创建。这个类提供了一种访问其唯一的对象的方式,可以直接访问,不需要实例化该类的对象。
单例模式有分为饿汉式和懒汉式
特点:
应用实例:
优点:
缺点:没有接口,不能继承,与单一职责原则冲突,一个类应该只关心内部逻辑,而不关心外面怎么样来实例化。
使用场景:
注意事项:getInstance() 方法中需要使用同步锁 synchronized (Singleton.class) 防止多线程同时进入造成 instance 被多次实例化。
代码
//单例模式-饿汉式
public class SingletonDemo {
public static void main(String[] args) {
//编译错误:无法实例化
// Singleton singleton = new Singleton();
//正确获取对象的方法
Singleton singleton = Singleton.getINSTANCE();
singleton.hello();
}
}
class Singleton{
//创建一个本身对象
private static final Singleton INSTANCE = new Singleton();
//让构造方法为private,这样该类就不会被实例化
private Singleton(){}
//创建一个获取对象的方法
public static Singleton getINSTANCE() {
return INSTANCE;
}
public void hello(){
System.out.println("Hello World! ——单例模式-饿汉式");
}
} 结果
Hello World! ——单例模式-饿汉式
这种方式是最基本的实现方式,这种实现最大的问题就是不支持多线程。因为没有加锁 synchronized,所以严格意义上它并不算单例模式。
这种方式 lazy loading 很明显,不要求线程安全,在多线程不能正常工作。
代码
//单例模式-懒汉式
public class SingletonDemo2 {
public static void main(String[] args) {
Singleton2 singleton = Singleton2.getInstance();
singleton.hello();
}
}
class Singleton2{
private static Singleton2 instance;
private Singleton2(){}
public static Singleton2 getInstance() {
if (instance == null){
instance = new Singleton2();
}
return instance;
}
public void hello(){
System.out.println("Hello World! ——单例模式-懒汉式");
}
}
结果
Hello World! ——单例模式-懒汉式
描述:这种方式具备很好的 lazy loading,能够在多线程中很好的工作,但是,效率很低,99% 情况下不需要同步。
优点:第一次调用才初始化,避免内存浪费。
缺点:必须加锁 synchronized 才能保证单例,但加锁会影响效率。
getInstance() 的性能对应用程序不是很关键(该方法使用不太频繁)。
代码
//单例模式-懒汉式
public class SingletonDemo3 {
public static void main(String[] args) {
Singleton3 singleton = Singleton3.getInstance();
singleton.hello();
}
}
class Singleton3{
private static Singleton3 instance;
private Singleton3(){}
public synchronized static Singleton3 getInstance() {
if (instance == null){
instance = new Singleton3();
}
return instance;
}
public void hello(){
System.out.println("Hello World! ——单例模式-懒汉式");
}
}
结果
Hello World! ——单例模式-懒汉式