Java-多线程(下)
线程的同步与死锁是多线程里面最需要重点理解的概念。这种操作的核心问题在于:每一个线程对象轮番强占资源
带来的问题。
关于窗口卖票问题,若不采用线程同步,当三个窗口同时开始卖票,因为多线程之间的资源是共享的,当三个窗口都获取到票仅剩一张的时候,三个窗口都满足可以卖票这一条件,故对票ticket做三次-1操作,则ticket的最终结果是-1;从而需要线程同步
synchronized处理同步问题
使用synchronized关键字处理有两种模式:同步代码块、同步方法
使用同步代码块 : 如果要使用同步代码块必须设置一个要锁定的对象,所以一般可以锁定当前对象:this。
synchronized(this){ ... }
使用同步方法:
在方法前加synchronized修饰
public synchronized void fun()
同步虽然可以保证数据的完整性(线程安全操作),但是其执行的速度会很慢。
a) 使用synchronized锁这个类对应的Class对象
synchronized(类名.class) {}
b) static synchronized方法,static方法可以直接类名加方法名调用,方法中无法使用this,所以它锁的不是this,而是 类的Class对象,所以,static synchronized方法也相当于全局锁,相当于锁住了代码段。
Lock锁的使用
1.在类中声明并实例化一个Lock类的属性 private Lock ticketLock = new ReentrantLock() ;
2.使用ticketLock.lock(){ ... } 实现同步代码
package www.bit.java.testdemo;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
class MyThread implements Runnable { private int ticket = 500; private Lock ticketLock = new ReentrantLock(); @Override public void run() { for (int i = 0; i < 500; i++) { ticketLock.lock(); try { if (this.ticket > 0) { // 还有票 try { Thread.sleep(20); } catch (InterruptedException e) { e.printStackTrace(); } // 模拟网络延迟 System.out.println(Thread.currentThread().getName() + ",还有" + this.ticket - -+" 张票"); } } finally { ticketLock.unlock(); } } }
}
public class TestDemo { public static void main(String[] args) { MyThread mt = new MyThread(); Thread t1 = new Thread(mt, "黄牛A"); Thread t2 = new Thread(mt, "黄牛B"); Thread t3 = new Thread(mt, "黄牛C"); t1.setPriority(Thread.MIN_PRIORITY); t2.setPriority(Thread.MAX_PRIORITY); t3.setPriority(Thread.MAX_PRIORITY); t1.start(); t2.start(); t3.start(); }
}
wait()方法 (wait()方法就是使线程停止运行)
1. 方法wait()的作用是使当前执行代码的线程进行等待,wait()方法是Object类的方法,该方法是用来将当前线 程置入“预执行队列”中,并且在wait()所在的代码处停止执行,直到接到通知或被中断为止。 2. wait()方法只能在同步方法中或同步块中调用。如果调用wait()时,没有持有适当的锁,会抛出异常。
3. wait()方法执行后,当前线程释放锁,线程与其它线程竞争重新获取锁。
notify()方法 (notify方法就是使停止的线程继续运行)
1. 方法notify()也要在同步方法或同步块中调用,该方法是用来通知那些可能等待该对象的对象锁的其它线程, 对其发出通知notify,并使它们重新获取该对象的对象锁。如果有多个线程等待,则有线程规划器随机挑选出 一个呈wait状态的线程。 2. 在notify()方法后,当前线程不会马上释放该对象锁,要等到执行notify()方法的线程将程序执行完,也就是退
出同步代码块之后才会释放对象锁。
notifyAll()方法
notify方法只是唤醒某一个等待线程,那么如果有多个线程都在等待中怎么办呢,这个时候就可以使用
notifyAll方法可以一次唤醒所有的等待线程。