前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >可重入锁详解(什么是可重入)

可重入锁详解(什么是可重入)

作者头像
全栈程序员站长
发布2022-08-03 11:23:18
7290
发布2022-08-03 11:23:18
举报
文章被收录于专栏:全栈程序员必看

大家好,又见面了,我是你们的朋友全栈君。

可重入锁详解

概述

什么是 “可重入”,可重入就是说某个线程已经获得某个锁,可以再次获取锁而不会出现死锁。例如

代码语言:javascript
复制
package com.test.reen;

// 演示可重入锁是什么意思,可重入,就是可以重复获取相同的锁,synchronized和ReentrantLock都是可重入的
// 可重入降低了编程复杂性
public class WhatReentrant { 
   
	public static void main(String[] args) { 
   
		new Thread(new Runnable() { 
   
			@Override
			public void run() { 
   
				synchronized (this) { 
   
					System.out.println("第1次获取锁,这个锁是:" + this);
					int index = 1;
					while (true) { 
   
						synchronized (this) { 
   
							System.out.println("第" + (++index) + "次获取锁,这个锁是:" + this);
						}
						if (index == 10) { 
   
							break;
						}
					}
				}
			}
		}).start();
	}
}
代码语言:javascript
复制
package com.test.reen;

import java.util.Random;
import java.util.concurrent.locks.ReentrantLock;

// 演示可重入锁是什么意思
public class WhatReentrant2 { 
   
	public static void main(String[] args) { 
   
		ReentrantLock lock = new ReentrantLock();
		
		new Thread(new Runnable() { 
   
			@Override
			public void run() { 
   
				try { 
   
					lock.lock();
					System.out.println("第1次获取锁,这个锁是:" + lock);

					int index = 1;
					while (true) { 
   
						try { 
   
							lock.lock();
							System.out.println("第" + (++index) + "次获取锁,这个锁是:" + lock);
							
							try { 
   
								Thread.sleep(new Random().nextInt(200));
							} catch (InterruptedException e) { 
   
								e.printStackTrace();
							}
							
							if (index == 10) { 
   
								break;
							}
						} finally { 
   
							lock.unlock();
						}

					}

				} finally { 
   
					lock.unlock();
				}
			}
		}).start();
	}
}

可以发现没发生死锁,可以多次获取相同的锁

可重入锁有

  • synchronized
  • ReentrantLock

使用ReentrantLock的注意点

ReentrantLock 和 synchronized 不一样,需要手动释放锁,所以使用 ReentrantLock的时候一定要手动释放锁,并且加锁次数和释放次数要一样

以下代码演示,加锁和释放次数不一样导致的死锁

代码语言:javascript
复制
package com.test.reen;

import java.util.Random;
import java.util.concurrent.locks.ReentrantLock;

public class WhatReentrant3 { 
   
	public static void main(String[] args) { 
   
		ReentrantLock lock = new ReentrantLock();
		
		new Thread(new Runnable() { 
   
			@Override
			public void run() { 
   
				try { 
   
					lock.lock();
					System.out.println("第1次获取锁,这个锁是:" + lock);

					int index = 1;
					while (true) { 
   
						try { 
   
							lock.lock();
							System.out.println("第" + (++index) + "次获取锁,这个锁是:" + lock);
							
							try { 
   
								Thread.sleep(new Random().nextInt(200));
							} catch (InterruptedException e) { 
   
								e.printStackTrace();
							}
							
							if (index == 10) { 
   
								break;
							}
						} finally { 
   
// lock.unlock();// 这里故意注释,实现加锁次数和释放次数不一样
						}

					}

				} finally { 
   
					lock.unlock();
				}
			}
		}).start();
		
		
		new Thread(new Runnable() { 
   
			
			@Override
			public void run() { 
   
				try { 
   
					lock.lock();
					
					for (int i = 0; i < 20; i++) { 
   
						System.out.println("threadName:" + Thread.currentThread().getName());
						try { 
   
							Thread.sleep(new Random().nextInt(200));
						} catch (InterruptedException e) { 
   
							e.printStackTrace();
						}
					}
				} finally { 
   
					lock.unlock();
				}
			}
		}).start();
		
		
	}
}

由于加锁次数和释放次数不一样,第二个线程始终无法获取到锁,导致一直在等待。

稍微改一下,在外层的finally里头释放9次,让加锁和释放次数一样,就没问题了

代码语言:javascript
复制
try { 
   
	lock.lock();
	System.out.println("第1次获取锁,这个锁是:" + lock);

	int index = 1;
	while (true) { 
   
		try { 
   
			lock.lock();
			System.out.println("第" + (++index) + "次获取锁,这个锁是:" + lock);
			
			... 代码省略节省篇幅...
		} finally { 
   
// lock.unlock();// 这里故意注释,实现加锁次数和释放次数不一样
		}

	}

} finally { 
   
	lock.unlock();
	
	// 在外层的finally里头释放9次,让加锁和释放次数一样,就没问题了
	for (int i = 0; i < 9; i++) { 
   
		lock.unlock();
		
	}
}

发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/124905.html原文链接:https://javaforall.cn

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2022年4月1,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 可重入锁详解
    • 概述
      • 使用ReentrantLock的注意点
      领券
      问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档