线程安全问题其实就是由多个线程同时处理共享资源所导致的。
要想解决线程安全问题,必须得保证下面用于处理共享资源的代码在任何时刻只能有一个线程访问。
为了实现这种限制, Java 中提供了同步机制。当多个线程使用同一个共享资源时,可以将处理共享资源的代码放置在一个代码块中,使用 synchronized 关键字来修饰,被称作同步代码块,其语法格式如下:
synchronized(lock){
操作共享资源代码
}
上面的代码中, lock 是—个锁对象,它是同步代码块的关键。当线程执行同步代码块时,首先会检查锁对象的标志位,默认情况下标志位为1,此时线程会执行同步代码块,同时将锁对象的标志位置为0。当一个新的线程执行到这段同步代码块时,由于锁对象的标志位为0,新线程会发生阻塞,等待当前线程执行完同步代码块后,锁对象的标志位被置为1,新线程才能进人同步代码块执行其中的代码。循环往复,直到共享资源被处理完为止。
这个过程就好比一个公用电话亭,只有前一个人打完电话出来后,后面的人才可以打。
ExampleM12:
// 同步代码块
//定义 Ticket1 类实现 Runnable 接口
class Ticket1 implements Runnable {
private int tickets = 10; //定义变量 tickets ,并赋值10
Object lock = new Object(); //定义任意对象,用作同步代码块的锁
@Override
public void run() {
while (true) {
synchronized (lock) { //定义同步代码块
try {
Thread.sleep(10); //经过休眠 10毫秒
}catch (InterruptedException e) {
e.printStackTrace();
}
if (tickets>0) {
System.out.println(Thread.currentThread().getName()+"---卖出的票"+tickets--);
}else { //如果 tickets 小于或等于0,跳出循环
break;
}
}
}
}
}
public class ExampleM12 {
public static void main(String[] args) {
Ticket1 ticket = new Ticket1(); //创建 Ticket1对象
//创建并开启四个线程
new Thread(ticket,"线程一").start();
new Thread(ticket,"线程二").start();
new Thread(ticket,"线程三").start();
new Thread(ticket,"线程四").start();
}
}
为了保证线程的持续执行,将同步代俏块取在死循环中,直到 ticket <=0时跳出循环。
注意 :同步代码块中的锁对象可以是任意类型的对象,但多个线程共享的锁对象必须是唯一的。“任意"说的是共享锁对象的类型。所以,锁对象的创建代码不能放到 run()方法中,否则每个线程运行到 run ()方法都会创建一个新对象,这样每个线程都会 有一个不同的锁,每个锁都有自已的标志位。线程之间便不能产生同步的效果。