实现多线程时,如果遇到共享资源的情况,需要对不同线程操作共享资源的代码块进行同步加锁:
package thread;
/*
通过分析,发现 RunnableDemo 会打印出0,-1,-2等错票。多线程的运行出现了安全问题。
问题的原因:
当多条语句在操作同一个线程共享数据时,一个线程对多条语句只执行了一部分,还没有执行完,
另一个线程参与进来执行。导致共享数据的错误。
解决办法:
对多条操作共享数据的语句,只能让一个线程都执行完。在执行过程中,其他线程不可以参与执行。
Java对于多线程的安全问题提供了专业的解决方式。就是同步代码块。
synchronized(对象)
{
需要被同步的代码(操作共享数据的代码段)
}
对象如同锁。持有锁的线程可以在同步中执行。没有持有锁的线程即使获取cpu的执行权,也进不去,因为没有获取锁。
火车上的卫生间---同步锁的经典例子。
同步的前提:
1,必须要有两个或者两个以上的线程。
2,必须是多个线程使用同一个锁。
必须保证同步中只能有一个线程在运行。
好处:解决了多线程的安全问题。
弊端:多个线程需要判断锁,较为消耗资源,
*/
public class Thread_3_SynDemo{
public static void main(String[] args) {
Ticket2 t = new Ticket2();
Thread t1 = new Thread(t);
Thread t2 = new Thread(t);
Thread t3 = new Thread(t);
Thread t4 = new Thread(t);
t1.start();
t2.start();
t3.start();
t4.start();
}
}
class Ticket2 implements Runnable{
private int tick = 100;
Object obj = new Object();
public void run(){
while(true){
//任何线程在这一步都会检查 obj 是否被锁(检查其标志位)
synchronized(obj){ //任何独一无二的东西都可以作为“锁”,如某个对象,某个类本身,只要各个线程持有的是同一个锁就可以
if(tick>0){
try{Thread.sleep(10);}catch(Exception e){}
System.out.println(Thread.currentThread().getName()+"....sale : "+ tick--);
}else{
break;
}
}
//同步代码块执行完毕之后obj对象的锁会被释放
}
}
}