线程唯一的单例

经典设计模式的单例模式是指进程唯一的对象实例,实现code如下:

package cun.zheng.weng.design.sinstnce;

import java.util.concurrent.CountDownLatch;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicLong;

public class IdGenerator {

    private IdGenerator(){}

    private AtomicLong generator = new AtomicLong(0);

    static class IdGeneratorHolder {
        private final static IdGenerator ID_GENERATOR = new IdGenerator();
    }

    public static IdGenerator getInstance(){
        return IdGeneratorHolder.ID_GENERATOR;
    }

    public Long getId(){
        return generator.getAndIncrement();
    }

    public static void main(String[] args) throws InterruptedException {
        ThreadPoolExecutor executor = new ThreadPoolExecutor(3,3,1000, TimeUnit.MILLISECONDS,new LinkedBlockingQueue<>());
        CountDownLatch latch = new CountDownLatch(10);
        for(int i=0;i<10;i++){
            Integer taskId = i;
            executor.submit(()->{
                Long threadId = Thread.currentThread().getId();
                System.out.println("Task "+ taskId +" generate id->" + IdGenerator.getInstance().getId() + " ,task done by "+ threadId);
                latch.countDown();
            });
        }
        latch.await();
        executor.shutdownNow();
    }
}

输出结果:
Task 0 generate id->0 ,task done by 9
Task 1 generate id->1 ,task done by 10
Task 2 generate id->2 ,task done by 11
Task 3 generate id->3 ,task done by 9
Task 5 generate id->5 ,task done by 10
Task 4 generate id->4 ,task done by 11
Task 7 generate id->7 ,task done by 10
Task 6 generate id->6 ,task done by 9
Task 9 generate id->9 ,task done by 10
Task 8 generate id->8 ,task done by 11

别跟我扯什么双重检测单例模式,性能太差,不考虑,单例模式就用静态内部类实现,高性能+延迟加载,还要什么双重检测!

那么,问题来了,如果我要实现线程绑定呢,换句话说就是实现线程间唯一的单例,进程中可以有多个实例。

思路很简单:将线程id绑定对象实例就行了。以下使用Map和JDK原生的ThreadLocal实现这个想法。

package cun.zheng.weng.design.sinstnce;

import java.util.Map;
import java.util.concurrent.*;
import java.util.concurrent.atomic.AtomicLong;

public class ThreadSingleInstance {

    private ThreadSingleInstance(){}

    AtomicLong atomicLong = new AtomicLong(0);

    private static final Map<Long,ThreadSingleInstance> idGeneratorHolder = new ConcurrentHashMap<>();

    public Long getId(){
        return atomicLong.getAndIncrement();
    }

    public static ThreadSingleInstance getInstance(){
        Long threadId = Thread.currentThread().getId();
        idGeneratorHolder.putIfAbsent(threadId,new ThreadSingleInstance());
        return idGeneratorHolder.get(threadId);
    }

    public static void main(String[] args) throws InterruptedException {
        ThreadPoolExecutor executor = new ThreadPoolExecutor(3,3,1000, TimeUnit.MILLISECONDS,new LinkedBlockingQueue<>());
        CountDownLatch latch = new CountDownLatch(10);
        for(int i=0;i<10;i++){
            Integer taskId = i;
            executor.submit(()->{
                Long threadId = Thread.currentThread().getId();
                System.out.println("Task "+ taskId +" generate id->" + ThreadSingleInstance.getInstance().getId() + " ,task done by "+ threadId);
                latch.countDown();
            });
        }
        latch.await();
        executor.shutdownNow();
    }
}

输出:
Task 1 generate id->0 ,task done by 10
Task 2 generate id->0 ,task done by 11
Task 3 generate id->1 ,task done by 10
Task 4 generate id->1 ,task done by 11
Task 5 generate id->2 ,task done by 10
Task 6 generate id->2 ,task done by 11
Task 7 generate id->3 ,task done by 10
Task 8 generate id->3 ,task done by 11
Task 9 generate id->4 ,task done by 10
Task 0 generate id->0 ,task done by 9

 

package cun.zheng.weng.design.sinstnce;

import java.util.concurrent.CountDownLatch;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicLong;

public class ThreadInstanceByThreadLocal {

    private ThreadInstanceByThreadLocal(AtomicLong atomicLong){
        this.atomicLong = atomicLong;
    }

    private AtomicLong atomicLong;

    public Long getId(){
        return atomicLong.getAndIncrement();
    }

    public static ThreadInstanceByThreadLocal getInstance(){
        return new ThreadInstanceByThreadLocal(new AtomicLong(0));
    }

    public static void main(String[] args) throws InterruptedException {
        ThreadPoolExecutor executor = new ThreadPoolExecutor(3,3,1000, TimeUnit.MILLISECONDS,new LinkedBlockingQueue<>());
        CountDownLatch latch = new CountDownLatch(10);
        ThreadLocal<ThreadInstanceByThreadLocal> threadLocal = new ThreadLocal<ThreadInstanceByThreadLocal>(){
            @Override
            protected ThreadInstanceByThreadLocal initialValue() {
                return ThreadInstanceByThreadLocal.getInstance();
            }
        };
        for(int i=0;i<10;i++){
            Integer taskId = i;
            executor.submit(()->{
                Long threadId = Thread.currentThread().getId();
                System.out.println("Task "+ taskId +" generate id->" + threadLocal.get().getId() + " ,task done by "+ threadId);
                latch.countDown();
            });
        }
        latch.await();
        executor.shutdownNow();
    }
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
线程安全单例模式是一种在多线程环境下确保只有一个实例被创建的设计模式。在实际使用中,有时需要保证某个类的对象在整个应用程序中只能有一个实例存在,并且该实例在任何时候都能被访问和使用。 线程安全单例模板的实现一般包含以下几个要点: 1. 构造器私有化:通过将类的构造函数私有化,可以防止在类外部直接实例化对象。 2. 静态成员对象:在类内部创建一个私有的静态成员对象,用于存储类的唯一实例。 3. 全局访问方法:通过提供一个公共的静态方法,来获取类的唯一实例。在该方法内部进行实例化操作,保证只有一个实例被创建。 4. 线程安全性:由于多线程环境下会有多个线程同时访问该方法,需要考虑线程安全问题。可以通过加锁机制,或者使用双重检查锁定(double-checked locking)来保证线程安全。 双重检查锁定是一种常用的实现方式,具体步骤如下: 1. 在全局访问方法内进行第一次判断,检查实例是否已经被创建。 2. 若实例为空,则进行同步锁定,防止其他线程同时进入。 3. 在同步块内再次检查实例是否为空,如果为空则进行实例化。 4. 返回实例。 这样可以确保在多线程环境下,只有一个实例被创建,同时保证访问的效率和线程安全性。 总之,线程安全单例模板是一种重要的设计模式,它可以保证在多线程环境下只有一个实例存在,并且正常进行访问。通过适当的加锁机制,可以确保线程安全性,保护对象的一致性和可靠性。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值