3、Lock锁(重点)
===========
3.1、传统synchronized
本质: 队列和锁,放在方法上锁的是this,放在代码块中锁的是()里面的对象
synchronized(obj){
}
3.2、Lock 接口
- reentrantLock构造器
非公平锁:NonfairSync()
十分不公平,可以插队(默认)
公平锁:FairSync()
十分公平,先来后到,一定要排队
public ReentrantLock() {
sync = new NonfairSync(); //无参默认非公平锁
}
public ReentrantLock(boolean fair) {
sync = fair ? new FairSync() : new NonfairSync();//传参为true为公平锁
}
3.3、synchronized 和 lock 区别
| 比较 | Lock | synchronized |
| — | — | — |
| 类型 | java接口 | 内置关键字 |
| 开启释放 | 显式锁手动 开启 和 关闭 | 隐式锁
出了作用域自动释放 |
| 代码锁 | 支持 | 支持 |
| 方法锁 | 不支持 | 支持 |
| 锁状态 | 可以判断是否获取到了锁 | 无法判断获取锁的状态 |
| 线程阻塞 | Lock锁就不一定会等待下去 | 线程1(获得锁,阻塞),线程2(等待) |
| 可重入性 | 可重入的,可以判断锁 | 可重入锁,不可以中断的 |
| 公平性 | 默认非公平的(可设置) | 非公平的 |
| 性能 | 性能更好
JVM将花费较少的时间来调度 | 性能一般
JVM将花费较多的时间来调度管理 |
| 扩展性 | 有更好的扩展性(提供更多的子类) | 不支持 |
| 适用场景 | 适合锁大量的同步代码 | 适合锁少量的代码同步问题 |
4、生产者和消费者问题
===========
面试高频:单例模式、八大排序、生产者消费者、死锁
4.1、synchronized实现
wait、notify 必须在 synchronizied 声明的代码中,否则会抛出异常 java.lang.IllegalMonitorStateException
package providerConsumer;
/**
-
@author ajun
-
Date 2021/7/3
-
@version 1.0
-
synchronized版生产者消费者
*/
public class Syn {
public static void main(String[] args) {
Data data = new Data();
//线程1
new Thread(() -> {
for (int i = 0; i < 5; i++) {
try {
data.increment();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},“A”).start();
//线程2
new Thread(() -> {
for (int i = 0; i < 5; i++) {
try {
data.decrement();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},“B”).start();
//线程3
new Thread(() -> {
for (int i = 0; i < 5; i++) {
try {
data.increment();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},“C”).start();
//线程4
new Thread(() -> {
for (int i = 0; i < 5; i++) {
try {
data.decrement();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},“D”).start();
}
}
class Data{
private int num;
public int getNum() {
return num;
}
//增加
public synchronized void increment() throws InterruptedException {
//判断等待
//用while,不用if,防止虚假唤醒
while(num!=0){
this.wait();
}
//业务代码
num++;
System.out.println(Thread.currentThread().getName() + " --> " + num);
//通知
this.notifyAll();
}
//减少
public synchronized void decrement() throws InterruptedException {
//判断等待
//用while,不用if,防止虚假唤醒
while(num == 0){
this.wait();
}
//业务代码
num–;
System.out.println(Thread.currentThread().getName() + " --> " + num);
//通知
this.notifyAll();
}
}
- 可能存在的问题
虚假唤醒
在判断等待时,如果用 if ,当线程多的时候,可能会有虚假唤醒
解决办法
if 判断改为 while 判断
因为 if 只会执行一次,执行完会接着向下执行 if()外边的
而 while 不会,直到条件满足才会向下执行 while()外边的
4.2、JUC实现
在JUC中,Lock 替换 synchronized,await 替换 wait,signal 替换 notify。(signal 信号)
package providerConsumer;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
/**
-
@author ajun
-
Date 2021/7/3
-
@version 1.0
*/
public class Loc {
public static void main(String[] args) {
Data2 data2 = new Data2();
new Thread(() -> {
for (int i = 0; i < 5; i++) {
try {
data2.increment();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},“A”).start();
new Thread(() -> {
for (int i = 0; i < 5; i++) {
try {
data2.decrement();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},“B”).start();
new Thread(() -> {
for (int i = 0; i < 5; i++) {
try {
data2.increment();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},“C”).start();
new Thread(() -> {
for (int i = 0; i < 5; i++) {
try {
data2.decrement();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
},“D”).start();
}
}
class Data2{
private int num = 0;
Lock lock = new ReentrantLock();//定义锁
Condition condition = lock.newCondition();//定义同步监视器
public int getNum() {
return num;
}
//增加
public void increment() throws InterruptedException {
lock.lock();//加锁
try {
//判断等待
while (num != 0){
condition.await();//等待
}
//业务代码
num++;
System.out.println(Thread.currentThread().getName() + " --> " + num);
//通知
condition.signalAll();
} finally {
lock.unlock();//解锁
}
}
//减少
public void decrement() throws InterruptedException {
lock.lock();//加锁
try {
//判断等待
while (num == 0){
condition.await();//等待
}
//业务代码
num–;
System.out.println(Thread.currentThread().getName() + " --> " + num);
//通知
condition.signalAll();
} finally {
lock.unlock();//解锁
}
}
}
Condition实现精准通知唤醒
5、8锁现象
======
8锁就是关于锁的八个现象
① 非静态同步方法的默认锁是 this,静态同步方法的默认锁是 class
② 某一时刻内,只能有一个线程有锁,无论几个方法;前提是用的同一把锁
参考:https://www.cnblogs.com/shamao/p/11045282.html
5.1、2线程 1对象 2方法
两个线程 调用 同一个对象 的 两个同步方法
package lock8;
/**
-
@author ajun
-
Date 2021/7/4
-
@version 1.0
-
两个线程调用同一个对象的两个同步方法
*/
public class Lock1 {
public static void main(String[] args) {
//同一对象
Number number = new Number();
//线程1
new Thread(() -> {number.getOne();},“A”).start();
//线程2
new Thread(() -> {number.getTwo();},“B”).start();
}
}
class Number{
//同步方法1 (非static)
public synchronized void getOne(){
System.out.println(Thread.currentThread().getName() + “: one”);
}
//同步方法2 (非static)
public synchronized void getTwo(){
System.out.println(Thread.currentThread().getName() + “: two”);
}
}
运行结果如下:
A: one
B: two
分析:
被 synchronized 修饰的方法,锁的对象是方法的调用者。因为两个方法的调用者是同一个,所以两个方法用的是同一个锁,先调用方法的先获得锁,先执行
5.2、2线程 1对象 2方法(1sleep)
新增sleep()给某个方法
TimeUnit.SECONDS.sleep(2);
分析:
不管在何处添加休眠后,会中途休眠,但不影响执行顺序。
被synchronized修饰的方法,锁的对象是方法的调用者。因为两个方法的调用者是同一个,所以两个方法用的是同一个锁,先调用方法的先获得锁,先执行,第二个方法只有在第一个方法执行完释放锁之后才执行
5.3、3线程 1对象 3方法(1普通)
新增一个线程调用 同一对象 新增的一个普通方法
分析:
新增的方法没有被 synchronized 修饰,不是同步方法,不受锁的影响,所以不需要等待。其他线程共用了一把锁,所以还需要等待。
5.4、2线程 2对象 2方法(1sleep)
两个线程 调用 两个对象 的同步方法,其中一个方法有sleep()
分析:
被 synchronized 修饰的方法,锁的对象是方法的调用者。因为用了两个对象调用各自的方法,所以两个方法的调用者不是同一个,所以两个方法用的不是同一个锁,后调用的方法不需要等待先调用的方法
5.5、2线程 1对象 2方法(1static)
sleep()的方法设置为static,并且让两个线程用同一个对象调用两个方法
分析:
被synchronized和static修饰的方法,锁的对象是类的class对象。仅仅被synchronized修饰的方法,锁的对象是方法的调用者。因为两个方法锁的对象不是同一个,所以两个方法用的不是同一个锁,后调用的方法不需要等待先调用的方法。
5.6、2线程 1对象 2方法(2sattic)
将两个方法均设置为static方法,并且让两个线程用同一个对象调用两个方法
分析:
被synchronized和static修饰的方法,锁的对象是类的class对象。因为两个同步方法都被static修饰了,所以两个方法用的是同一个锁,后调用的方法需要等待先调用的方法
5.7、2线程 2对象 2方法(1static)
将两个方法中有sleep()的方法设置为static方法,另一个方法去掉static修饰,让两个线程用 两个对象 调用两个方法
分析:
被synchronized和static修饰的方法,锁的对象是类的class对象。仅仅被synchronized修饰的方法,锁的对象是方法的调用者。即便是用同一个对象调用两个方法,锁的对象也不是同一个,所以两个方法用的不是同一个锁,后调用的方法不需要等待先调用的方法。
5.8、2线程 2对象 2方法(2static)
将两个方法均设置为static方法,并且让两个线程用 一个对象 调用两个方法
一、网安学习成长路线图
网安所有方向的技术点做的整理,形成各个领域的知识点汇总,它的用处就在于,你可以按照上面的知识点去找对应的学习资源,保证自己学得较为全面。
二、网安视频合集
观看零基础学习视频,看视频学习是最快捷也是最有效果的方式,跟着视频中老师的思路,从基础到深入,还是很容易入门的。
三、精品网安学习书籍
当我学到一定基础,有自己的理解能力的时候,会去阅读一些前辈整理的书籍或者手写的笔记资料,这些笔记详细记载了他们对一些技术点的理解,这些理解是比较独到,可以学到不一样的思路。
四、网络安全源码合集+工具包
光学理论是没用的,要学会跟着一起敲,要动手实操,才能将自己的所学运用到实际当中去,这时候可以搞点实战案例来学习。
五、网络安全面试题
最后就是大家最关心的网络安全面试题板块
网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!