多线程程序出现安全问题的原因:
A:是多线程程序。
B:有共享数据。
C:针对共享数据有多条语句操作。现在的程序存在着问题,所以,我们要改进该程序,让它没有问题。
怎么改进呢?
只需要把多线程环境中,操作共享数据的操作给变成单线程的就没有问题了。-
Java针对这种情况,就提供了同步技术:同步代码块。
格式:
synchronized(对象) {
需要被同步的代码。
}A:对象? 如果不知道用哪个对象,就用Object对象。 B:需要被同步的代码? 哪些代码导致出现了问题,就把哪些代码给同步起来。 哪些代码会出问题呢? 有共享数据。 针对共享数据有多条语句操作。
加入同步后,居然还有问题,为什么呢?
同步代码块中的对象针对多个线程必须是同一个。
其实这个对象被称为同步锁对象。同步代码块的锁对象是谁?
任意对象同步方法的锁对象是谁?
this对象静态方法的锁对象是谁?
类.class 字节码文件对象
当前类的字节码文件代码如下
public static void main(String[] args) {
Ticket t = new Ticket();
Thread t1 = new Thread(t, "窗口1");
Thread t2 = new Thread(t, "窗口2");
Thread t3 = new Thread(t, "窗口3");
Thread t4 = new Thread(t, "窗口4");
t1.start();
t2.start();
t3.start();
t4.start();
}
Ticket的类 ----- 同步锁
public class Ticket implements Runnable {
private int tickets = 100;
private Object obj = new Object();
@Override
public void run() {
while (true) {
// t1,t2,t3,t4
// tickets=1;
// 假设t1过来,看到synchronized,就知道下面的代码被同步了。
// 看到obj就知道这是锁对象。假设t1进去前,obj这个锁对象表示开的状态。
synchronized (obj) {
// t1进来,就把锁对象的状态改为关的状态。
if (tickets > 0) {
try {
Thread.sleep(10);// t1睡下了
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()
+ "正在出售第" + (tickets--) + "张票");
}
}
// t1出来了,你们继续抢,我也还可以抢。
}
}
}
Ticket的类 ----- Lock锁
public class Ticket implements Runnable {
public int ticks = 100;
//定义lock锁对象
private Lock lock = new ReentrantLock();
@Override
public void run() {
lock.lock();
if (ticks > 0) {
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + "正在出售第" + (ticks--) + "张票");
}
//释放锁
lock.unlock();
}
}