设计模式之单例模式

单例模式

单例模式(Singleton Pattern)是 Java 中最简单的设计模式之一。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。

这种模式涉及到一个单一的类,该类负责创建自己的对象,同时确保只有单个对象被创建。这个类提供了一种访问其唯一的对象的方式,可以直接访问,不需要实例化该类的对象。

注意:

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

单例模式分为懒汉式和饿汉式

饿汉式

所谓饿汉式是指,无论在程序运行期间是否需要该对象,都会提前创建一个单例对象。该方式存在的缺点,创建的对象可能在程序运行期间没有被使用过,占用了堆内存空间。

package com.xxx.sigleton;

/**
 * @packageName: com.xxx.sigleton
 * @user: andyliu
 * @date: 2023/4/15 10:06
 * @email 115176513@qq.com
 * @description: TODO
 */

public class HungrySingleton {
    private static HungrySingleton singleton = new HungrySingleton();
    private HungrySingleton(){

    }
    public static HungrySingleton getInstance(){
        return singleton;
    }

    public static void main(String[] args) {
        HungrySingleton instance = HungrySingleton.getInstance();
        HungrySingleton instance2 = HungrySingleton.getInstance();
        System.out.println(instance==instance2);
    }
}

原理

采用private构造,通过静态成员变量只能加载一次的原理,实现单例模式。

线程安全测试

package com.xxx.sigleton;

import java.util.concurrent.Callable;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;

/**
 * @packageName: com.xxx.sigleton
 * @user: andyliu
 * @date: 2023/4/15 10:09
 * @email 115176513@qq.com
 * @description: TODO
 */

public class TestHungrySingleton {
    public static void main(String[] args) throws Exception {
        ExecutorService service = Executors.newFixedThreadPool(3);
        Future<HungrySingleton> submit = service.submit(new HungryCallable());
        Future<HungrySingleton> submit2 = service.submit(new HungryCallable());

        HungrySingleton hungrySingleton = submit.get();
        HungrySingleton hungrySingleton2 = submit2.get();
        System.out.println(hungrySingleton==hungrySingleton2);
    }
    static class HungryCallable implements Callable<HungrySingleton>{
        @Override
        public HungrySingleton call() throws Exception {
            return HungrySingleton.getInstance();
        }
    }
}


  • 枚举
package com.xxx.sigleton;

/**
 * @packageName: com.xxx.sigleton
 * @user: andyliu
 * @date: 2023/4/15 15:20
 * @email 115176513@qq.com
 * @description: TODO
 */

public enum EnumSingleton {
    INSTANCE

}


懒汉式

私有构造方式

package com.xxx.sigleton;

/**
 * @packageName: com.xxx.sigleton
 * @user: andyliu
 * @date: 2023/4/15 10:21
 * @email 115176513@qq.com
 * @description: TODO
 */

public class SingletonObject {
    private static SingletonObject instance;
    private SingletonObject(){

    }
    public static SingletonObject getInstance(){
        if(instance==null){
            instance = new SingletonObject();
        }
        return instance;
    }

    public static void main(String[] args) {
        SingletonObject object = SingletonObject.getInstance();
        SingletonObject object2 = SingletonObject.getInstance();
        System.out.println(object==object2);
    }
}

多线程测试

package com.xxx.sigleton;

import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;

/**
 * @packageName: com.xxx.sigleton
 * @user: andyliu
 * @date: 2023/4/15 10:21
 * @email 115176513@qq.com
 * @description: TODO
 */

public class SingletonObject {
    private static SingletonObject instance;
    private SingletonObject(){

    }
    public static SingletonObject getInstance(){
        try {
            Thread.sleep(100);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        if(instance!=null){

        }else{
            instance = new SingletonObject();

        }
        return instance;
    }

    public static void main(String[] args) throws ExecutionException, InterruptedException {
        Thread thread = new Thread(() -> {
            SingletonObject instance = SingletonObject.getInstance();
            System.out.println(Thread.currentThread().getName() + ":" + instance.hashCode());
        });
        Thread thread2 = new Thread(() -> {
            SingletonObject instance = SingletonObject.getInstance();
            System.out.println(Thread.currentThread().getName() + ":" + instance.hashCode());
        });
        thread.start();
        thread2.start();

    }
}

Thread-0:786680875
Thread-1:641158370

Process finished with exit code 0

同步锁

package com.xxx.sigleton;

import java.util.concurrent.ExecutionException;

/**
 * @packageName: com.xxx.sigleton
 * @user: andyliu
 * @date: 2023/4/15 10:21
 * @email 115176513@qq.com
 * @description: TODO
 */

public class SingletonObject2 {
    private static SingletonObject2 instance;
    private SingletonObject2(){

    }
    public static synchronized SingletonObject2 getInstance(){
        try {
            Thread.sleep(100);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        if(instance!=null){

        }else{
            instance = new SingletonObject2();

        }
        return instance;
    }

    public static void main(String[] args) throws ExecutionException, InterruptedException {
        Thread thread = new Thread(() -> {
            SingletonObject2 instance = SingletonObject2.getInstance();
            System.out.println(Thread.currentThread().getName() + ":" + instance.hashCode());
        });
        Thread thread2 = new Thread(() -> {
            SingletonObject2 instance = SingletonObject2.getInstance();
            System.out.println(Thread.currentThread().getName() + ":" + instance.hashCode());
        });
        thread.start();
        thread2.start();

    }
}

双重检查锁

package com.xxx.sigleton;

import java.util.concurrent.ExecutionException;

/**
 * @packageName: com.xxx.sigleton
 * @user: andyliu
 * @date: 2023/4/15 10:21
 * @email 115176513@qq.com
 * @description: TODO
 */

public class SingletonObject2 {
    private static volatile SingletonObject2 instance;
    private SingletonObject2(){
        try {
            Thread.sleep(100);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
    public static  SingletonObject2 getInstance(){
        if(instance==null){
            synchronized (SingletonObject2.class){
                if(instance==null){
                    // 不是原子操作
                    instance = new SingletonObject2();
                }
            }
        }
        return instance;
    }

    public static void main(String[] args) throws ExecutionException, InterruptedException {
        Thread thread = new Thread(() -> {
            SingletonObject2 instance = SingletonObject2.getInstance();
            System.out.println(Thread.currentThread().getName() + ":" + instance.hashCode());
        });
        Thread thread2 = new Thread(() -> {
            SingletonObject2 instance = SingletonObject2.getInstance();
            System.out.println(Thread.currentThread().getName() + ":" + instance.hashCode());
        });
        thread.start();
        thread2.start();

    }
}

静态内部类

package com.xxx.sigleton;

/**
 * @packageName: com.xxx.sigleton
 * @user: andyliu
 * @date: 2023/4/15 15:06
 * @email 115176513@qq.com
 * @description: TODO
 */

public class SingletonObject3 {

    private SingletonObject3(){

    }
    static class InnerClass{
        private static SingletonObject3 instance=new SingletonObject3();
        public static SingletonObject3 getInstance(){
            return InnerClass.instance;
        }
    }

    public static void main(String[] args) {
        Thread t1 = new Thread(()->{
            SingletonObject3 obj = SingletonObject3.InnerClass.getInstance();
            System.out.println(Thread.currentThread().getName()+":"+obj.hashCode());
        });
        Thread t2 = new Thread(()->{
            SingletonObject3 obj = SingletonObject3.InnerClass.getInstance();
            System.out.println(Thread.currentThread().getName()+":"+obj.hashCode());
        });
        t1.start();
        t2.start();
    }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

_AndyLau

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

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

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

打赏作者

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

抵扣说明:

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

余额充值