当递归遇到synchronized

面试题:有一个synchronized方法,加入该方法发生递归调用,会导致线程死锁码?

解析:
  所谓递归函数就是自调用函数,在函数体内直接或间接的调用自己,即函数的嵌套是函数本身。
  递归方式有两种:直接递归和间接递归,直接递归就是在函数中出现调用函数本身。间接递归,指函数中调用了其他函数,而该其他函数又调用了本函数。
  那什么时候使用递归呢?一般来说当你要在某段代码逻辑中使用循环迭代的时候但是迭代的次数在迭代之前无法知晓的情况下使用递归。打个比方你要在一个文件夹中查找某个文件,
而这个文件夹底下有N多子文件夹和文件,当你在不知道有多少层文件夹和文件的情况下你就得用到递归了。
  递归的优点就是让代码显得很简洁,同时有些应用场景不得不使用递归比如前面说的找文件。递归是个好东西但是在某些时候也会给你带来一些麻烦。比如在多线程的环境下使用递归,
遇到了多线程那么就不得不面对同步的问题。而递归程序遇到同步的时候很容易出问题。多线程的递归就是指递归链中的某个方法由另外一个线程来操作,以下代码的意思都是这个意思即
调用recursive()和businessLogic()并非一个线程(如果是在一个线程中就不存在死锁问题,例如下面的recursive变成private就不存在问题。)

  
  public class Test {
  public void recursive(){
  this.businessLogic();
  }
  public synchronized void businessLogic(){
  System.out.println("处理业务逻辑");
  System.out.println("保存到数据库");
  this.recursive();
  }
  }
  

  

  以上这段代码就是个能形成死锁的代码,事实上这个“synchronized”放在“businessLogic()”和“recursive()”都会形成死锁,并且是多线程的情况下就会锁住!
他的逻辑顺序是先执行recursive()方法然后接下来执行businessLogic()方法同时将businessLogic()方法锁住,接下来程序进入businessLogic()方法内部执行完
打印语句后开始执行recursive(),进入recursive()后准备执行businessLogic(),等等问题来了!之前执行的businessLogic()的锁还没有放开这次又执行到这里了,
当然是过不去的了,形成了死锁!从这个例子我们总结出来一个规律就是在递归的时候在递归链上面的方法上加锁肯定会出现死锁(所谓递归链就是指recursive()链向businessLogic(),
businessLogic()又链回recursive()),解决这个问题的方法就是避免在递归链上加锁,请看以下的例子

  
  public class Test {
  public void recursive(){
  this.businessLogic();
  }
  public void businessLogic(){
  System.out.println("处理业务逻辑");
  this.saveToDB();
  this.recursive();
  }
  public synchronized void saveToDB(){
  System.out.println("保存到数据库");
  }
  }
  

  saveToDB()不在这条递归链上面自然不会出现死锁,所以说在递归中加锁是件很危险的事情,实在逃不过要加锁就加在最小的粒度的程序代码上以减小死锁的概率。

转载于:https://www.cnblogs.com/rgever/p/8795661.html

`@synchronized` 是 Objective-C 中用于实现线程安全的关键字。它的底层实现是通过互斥锁(mutex)来保证代码块的原子性执行。具体原理如下: 1. 当编译器遇到 `@synchronized` 关键字时,它将关键字后面的对象作为锁对象,并生成一段互斥锁的代码。 2. 在运行时,当某个线程进入 `@synchronized` 代码块时,它尝试获取锁对象的互斥锁。如果该锁对象的互斥锁已经被其他线程持有,则当前线程被阻塞,直到互斥锁被释放。 3. 当某个线程持有锁对象的互斥锁时,它可以执行 `@synchronized` 代码块中的代码。其他线程在等待互斥锁时被阻塞。 4. 当持有锁对象的线程执行完 `@synchronized` 代码块中的代码后,释放互斥锁,允许其他线程获取互斥锁并执行对应的代码。 通过使用互斥锁,`@synchronized` 关键字确保了同一时间只有一个线程可以进入被保护的代码块,从而避免了多个线程同时访问共享资源导致的数据竞争和不一致性。 需要注意的是,`@synchronized` 关键字在底层使用的是递归锁(recursive lock),这意味着同一个线程可以多次获取同一个锁对象的互斥锁,而不造成死锁。 然而,`@synchronized` 关键字在性能上可能不如其他更底层的锁机制(如 NSLock、pthread_mutex_t)高效。因此,在需要高性能的场景下,可以考虑使用其他更适合的锁机制来实现线程安全。 希望以上解答对你有所帮助!如果还有其他问题,请随时提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值