334 同步方法解决数据安全问题
(myThread)
【同步方法】
同步方法,就是把synchronized关键字加到方法声明上。
格式
修饰符 synchronized 返回值类型 方法名(参数){...} synchronized (obj) |
【同步方法的锁对象】
this
【static静态同步方法的锁对象】
类名.class,例如:synchronized (SellTicket.class)
> 非静态同步方法有什么存在的意义?
【代码思路】
定义一个int变量x,初始值0,运行一次后+1
把线程代码复制2份放在在if...else循环里,第偶数次运行进if循环,第奇数次运行进else循环。
分析:这样,同一时间允许2个线程同时运行
写一个sellTicket方法,把本来在else里的线程代码放进这个方法,else里清空,只写一行:调用sellTicket方法
现在的代码是这样的
private void sellTicket() {
synchronized (obj) {
if (ticketN > 0) {
...
}
}
}
在以上代码中,锁位于方法体内部,这就涉及到同步方法。
代码改成如下
private synchronized void sellTicket() {
if (ticketN > 0) {
...
}
}
运行,出问题了,又出了重复的票,这是因为 sellTicket没有设定obj对象。
但写成private synchronized (obj) void sellTicket() 会抛出编译时异常。这就涉及到同步方法的锁对象:this。要在同步方法上设定对象,写this就好了,this指代当前的对象。
把sellTicket()改成private static void sellTicket(),随之把int变量ticketN改成静态的private static int ticketN = 20;
运行,还是有重复的票。
正确做法是把对象写成类名.class(该类的字节码对象,后面在反射里讲一下),如下
if (x%2==0){
synchronized (SellTicket.class) {...}
else{...}
}
---------------------------------------------
package e334;
public class SellTicket implements Runnable {
private static int ticketN = 100;
private Object obj = new Object();
private int x = 0;
@Override
public void run() {
while (true) {
if (x % 2 == 0) {
synchronized (SellTicket.class) {
if (ticketN > 0) {
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + "the NO. " + ticketN + " ticket is selling.");
ticketN--;
}
}
} else {
sellTicket();
}
x++;
}
}
private static synchronized(SellTicket.class) void sellTicket() {
if (ticketN > 0) {
try {
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName() + "the NO. " + ticketN + " ticket is selling.");
ticketN--;
}
}
}
---------------------------------------------
package e334;
public class SellTicketDemo {
//略,与前节同
}