Java用户锁-针对不同用户不同的锁

场景:用户点赞、取消点赞。
问题:避免点赞与取消点赞冲突,在不考虑其他方法前提下,采用JAVA synchronized 进行同步锁操作,如果直接锁方法,那A用户点赞、取消点赞时,会导致其他用户无法点赞、取消点赞。所以采用锁用户ID的形式

    public static void main(String[] args) {
//        System.out.println(DigestUtils.md5DigestAsHex(("20210928#888" + "1010").getBytes()));
        Test test = new Test();
        new Thread(() -> {
            test.asd(1, 1);
        }).start();
        new Thread(() -> {
            test.asd(2, 2);
        }).start();
    }

    // id锁
    private static final Map<Integer, Object> locks = new HashMap<>();
    // 用完后,如果直接删除id锁,那如果此时id锁已被使用,同步会失效,所以要用下面这个中间件,判断是否还有ID的锁在用
    private static final Map<Integer, AtomicInteger> waitingLocks = new HashMap<>();

    private void asd(Integer i, Integer id) {
        System.out.println(i + ":before");
        Object lock;
        synchronized (locks){
            lock = locks.computeIfAbsent(id, j -> new Object());
            waitingLocks.computeIfAbsent(id, j -> new AtomicInteger()).incrementAndGet();
        }
        synchronized (lock) {
            System.out.println(i + ":in");
            try {
                Thread.sleep(2000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(i + ":end");
            // 用完删除
            synchronized (locks){
                if(waitingLocks.get(id).decrementAndGet() == 0){
                    locks.remove(id);
                    waitingLocks.remove(id);
                }
            }
            System.out.println(JSON.toJSON(locks));
            System.out.println(JSON.toJSON(waitingLocks));
        }
        System.out.println(i + ":after");
        System.out.println("");
    }

 

posted @ 2023-02-20 10:22  甜8筒  阅读(198)  评论(0编辑  收藏  举报