同步代码块
解决线程安全问题的一种方案:使用同步代码块
格式:synchronized(锁对象){
可能会出现线程安全问题的代码(访问共享数据的代码)}
注意:
1.通过代码块中的锁对象,可以使用任意的对象
2.但是必须保证多个线程使用的锁对象是同一个
3.锁对象作用:
把同步代码块锁住,只让一个线程在同步代码块中执行
1 | public class TestDesp implements Runnable{ |
同步方法
使用synchronized修饰的方法就叫做同步方法,保证A线程执行该方法的时候,其他线程只能在方法外等着。
格式:
1 | public synchronized void method(){ |
使用步骤:
1.把访问了共享数据的代码抽取出来,放到一个方法中
2.在方法上添加synchronized修饰符;
格式:定义方法的格式
修饰符 synchronized 返回值类型 方法名(参数列表){
可能会出现线程安全的代码
}
1 | package Day002; |
静态同步
1 | package Day002; |
lock锁
解决线程安全问题的第三种方案:使用lock锁
java.util.concurrent.locks.lock锁
Lock 实现提供了比使用synchronized 方法和语句可获得更广泛的锁定操作。
Lock 接口中的方法:
void Lock() 获取锁
void unLock()释放锁
使用步骤:
1.在成员位置创建一个ReentrantLock对象
2.在可能会出现安全问题的代码前调用Lock接口中的方法Lock获取
package Day002;
import java.lang.*;
public class TestDesp implements Runnable{
private static int r = 100;
Object object = new Object();
@Override
//设置线程任务
public void run(){
//使用死循环,让卖票操作重复执行
while(true){
plytak();
}
}
}
/*
静态同步方法的
锁对象是谁?
不是this
this是创建对象后才产生的,静态方法优先于对象。
就是本类的class属性---class文件对象(反射)
也就是this
*/
public static synchronized void plytak (){
//先判断票是否存在
if(r>0){
//提高安全问题出现问题的概率,让程序睡眠
try {
Thread.sleep(10);
System.out.println(Thread.currentThread().getName()+"卖出了多少票"+r);
r--;
} catch (InterruptedException e) {
e.printStackTrace();
}finally{
lock.unlock();
}
}
}
}
1 | public static void main(String[] args) { |
线程状态
等待唤醒
线程之间的通信
案例:
1 | public class DemowaitAndNotify(){ |
wait带参方法
1 | package Day002; |
notifyAll方法
1 | package Day002; |