裁员在家,没有面试机会,整理整理面试知识点吧!
不得不知道的java 锁
Java 中,提供了两种方式来实现同步互斥访问(也就是锁):synchronized 和 Lock
多线程编程中,有可能会出现多个线程同时访问同一个共享、可变资源的情况,这个资源我们称之其为临界资源;
静态方法加锁:锁的是当前类对象
静态代码块:锁的是代码块里的对象
锁粗话 StringBubber .append 线程安全,多次append 就会进行一个锁 锁粗话
锁清除 synchronized (new object()) ,因为每次都是锁新对象,jvm 认为没有意思 所以不会加锁
逃逸分析:如果一个对象只有在一个线程栈中用到,没有其他线程用到,这样这个对象就会在一部分会在栈中不放到堆里。
unsafe.compareAndSwapInt(this, stateOffset, expect, update); setExclusiveOwnerThread(Thread.currentThread());
2 如果期望值跟内存之不相等,则获取共享变量判断是否等0 如果等于,则说明没有加锁,通过cas 继续加锁,如果不能与获取设置的线程,跟当前线程比对是否相等,如果相等则说明相同线程加锁,可以继续加锁,重入锁。
3 如果既不能加锁,也不是重入锁。则把当前线程放入队列里
用大白话来说,AQS就是基于CLH队列,用volatile修饰共享变量state,线程通过CAS去改变状态符,成功则获取锁成功,失败则进入等待队列,等待被唤醒。
ReentrantLock lock = new ReentrantLock(false);//false为非公平锁 lock.lock(); lock.tryLock(30, TimeUnit.SECONDS);// 30秒没加上锁,自动释放 lock.lockInterruptibly();// 锁中断
Condition condition = lock.newCondition(); //lock 等待通知
condition.await();
condition.signal();
可以替换 synchronized wait notify
obj.wait(); obj.notify(); 在synchronized 里否则报错
死锁:a线程锁定一个资源,同时想获取b线程的资源,b线程锁定一个资源,同时想获取a线程的资源。
java 死锁必要条件
1 互斥条件:资源不能被共享,只能由一个线程使用(threadLocal)
2 请求保持条件:一个线程因请求其他资源而阻塞时,对目前拥有的资源保持不放。(一次性请求所有资源)
3 不可抢占条件:进程已获得的资源,在未使用完之前,不能被剥夺( 超时500毫秒 ,自动释放锁) lock.tryLock(timeout)。
4 相互等待条件:t1 修改下单状态,减库存, t2减库存,下单 相互持有对方的锁 等待,死锁(逻辑顺序一致)
只要打破上面任意一个条件,即不会死锁
什么情况下会释放synchronized
执行完成会释放锁
抛异常会释放锁
wait 释放当前锁
return 释放锁