理解java中【同步】和【死锁】

时间:2023-03-09 15:37:47
理解java中【同步】和【死锁】

一.理解同步

要想解决资源共享的同步操作问题,可以使用两种方法:

  1. 使用同步代码块

之前学习过程中,代码块分为四种:

l         普通代码块:是直接定义在方法之中的;

l         构造块:是直接定义在类中的,优先于构造方法执行,会重复调用;

l         静态块:是使用static关键字声明的,优先于构造块执行,并且只执行一次;

l         同步代码块:是使用synchronized关键字声明的代码块,称为同步代码块

同步的时候必须指明同步的对象,一般情况下会将当前对象作为同步的对象,使用this关键字表示。 注意,同步会使程序运行变慢!

代码如下:

package cn.test.java.mutilthread;

class SyncThread2 implements Runnable{

private int ticket = 10;

public void run(){

for(int i = 0;i>10;i++){

synchronized (this) {

if(this.ticket>0){

try {

Thread.sleep(1000);

} catch (InterruptedException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

System.out.println(Thread.currentThread().getName()+"剩下票数:"+ticket--);

}

}

}

}

}

public class SyncThreadDemo2 {

public static void main(String[] args) {

SyncThread2 t = new SyncThread2();

Thread t1 = new Thread(t);

Thread t2 = new Thread(t);

Thread t3 = new Thread(t);

t1.start();

t2.start();

t3.start();

}

}

  1. 使用同步方法

package cn.test.java.mutilthread;

class SyncThread3 implements Runnable{

private int ticket = 10;

public void run(){

for(int i = 0;i>10;i++){

this.sale();// 调用同步方法

}

}

public synchronized void sale(){ //声明同步方法

if(this.ticket>0){

try {

Thread.sleep(1000);

} catch (InterruptedException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

System.out.println(Thread.currentThread().getName()+"剩下票数:"+ticket--);

}

}

}

public class SyncThreadDemo3 {

public static void main(String[] args) {

SyncThread3 t = new SyncThread3();

Thread t1 = new Thread(t);

Thread t2 = new Thread(t);

Thread t3 = new Thread(t);

t1.start();

t2.start();

t3.start();

}

}

二.理解死锁

死锁的出现是因为多个线程占用资源之后,没有进行释放,导致其他线程一直处于等待状态,在我们在代码中,出现死锁的原因有很多,极大多数是因为我们代码中编成的同步代码过多导致死锁的,

注意:使用synchronized同步代码中嵌入synchronized代码,非常容易导致死锁的出现。

代码如下:

class Zhangsan{     // 定义张三类

public void say(){

System.out.println("张三对李四说:“你给我画,我就把书给你。”") ;

}

public void get(){

System.out.println("张三得到画了。") ;

}

};

class Lisi{      // 定义李四类

public void say(){

System.out.println("李四对张三说:“你给我书,我就把画给你”") ;

}

public void get(){

System.out.println("李四得到书了。") ;

}

};

public class ThreadDeadLock implements Runnable{

private static Zhangsan zs = new Zhangsan() ;            // 实例化static型对象

private static Lisi ls = new Lisi() ;        // 实例化static型对象

private boolean flag = false ;  // 声明标志位,判断那个先说话

public void run(){   // 覆写run()方法

if(flag){

synchronized(zs){  // 同步张三

zs.say() ;

try{

Thread.sleep(500) ;

}catch(InterruptedException e){

e.printStackTrace() ;

}

synchronized(ls){

zs.get() ;

}

}

}else{

synchronized(ls){

ls.say() ;

try{

Thread.sleep(500) ;

}catch(InterruptedException e){

e.printStackTrace() ;

}

synchronized(zs){

ls.get() ;

}

}

}

}

public static void main(String args[]){

ThreadDeadLock t1 = new ThreadDeadLock() ;         // 控制张三

ThreadDeadLock t2 = new ThreadDeadLock() ;         // 控制李四

t1.flag = true ;

t2.flag = false ;

Thread thA = new Thread(t1) ;

Thread thB = new Thread(t2) ;

thA.start() ;

thB.start() ;

}

};

三.总结

1.多个线程在访问同一资源的时候需要进行同步操作。

2.同步使用synchronized关键字完成,分为同步代码块及同步方法。

3.过多的同步有可能造成死锁的产生,死锁是在程序运行时的一种状态,了解就行了。

4.想停止线程,在代码中设置标志位flag,利用标志位来控制线程在生命周期。