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

可重入锁详解(什么是可重入)可重入锁详解概述什么是“可重入”,可重入就是说某个线程已经获得某个锁,可以再次获取锁而不会出现死锁。例如packagecom.test.reen;//演示可重入锁是什么意思,可重入,就是可以重复获取相同的锁,synchronized和ReentrantLock都是可重入的//可重入降低了编程复杂性publicclassWhatReentrant{ publicsta…

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

可重入锁详解

概述

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

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();
	}
}
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的时候一定要手动释放锁,并且加锁次数和释放次数要一样

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

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次,让加锁和释放次数一样,就没问题了

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.net/124905.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • bs模型与cs模型共同点_bs模型的基本假设

    bs模型与cs模型共同点_bs模型的基本假设bs模式客户端通过浏览器,浏览web服务器上的网页,这样的模型叫bs模型,b指客户端browser,s指服务端server。在客户端和浏览器端之间走的报文是http协议(即超文本传输协议) cs模型客户端(client)发报文,服务器(server)收报文,服务器收到报文之后处理。这与bs模式没有很大区别,只不过是c与s间可以自定义数据传送报文。cs模式一般走的协议是tcp协议

    2022年9月14日
    0
  • PHP cURL 超时设置 CURLOPT_CONNECTTIMEOUT 和 CURLOPT_TIMEOUT 的区别[通俗易懂]

    PHP cURL 超时设置 CURLOPT_CONNECTTIMEOUT 和 CURLOPT_TIMEOUT 的区别

    2022年2月10日
    44
  • angular面试题及答案_angular面试

    angular面试题及答案_angular面试1.生命周期钩子生命周期的顺序,见下图:ngOnChanges:当组件数据绑定的输入属性发生变化是触发,该方法接收一个SimpleChanges对象,包括当前值和上一个属性值。首次调用一定发生在ngOnInit前,值得注意的是该方法仅限于对象的引用发生变化时才会触发。 ngOninit:初始化指令或组件,在angular第一次显示展示组件的绑定属性后调用,该方法只会调用一次 ng…

    2022年10月17日
    2
  • android调用相册并显示图片_Android获取相册列表

    android调用相册并显示图片_Android获取相册列表从之前的项目摘出来这段代码,去掉了裁剪功能packagecom.example.one;importandroid.content.Intent;importandroid.database.Cursor;importandroid.graphics.Bitmap;importandroid.graphics.BitmapFactory;importandroid.g…

    2022年9月15日
    1
  • C语言中的sizeof()和strlen()的区别[通俗易懂]

    C语言中的sizeof()和strlen()的区别sizeof()和strlen()经常会被初学者混淆,但其中有有很大区别:1.sizeof()【操作数所占空间的字节数大小】是一种c中的基本运算符。可以以类型、指针、数组和函数等作为参数。头文件类型为unsignedint。运算值在编译的时候就出结果,所以可以用来定义数组维数。chara[5]=”123″;intb=sizeof(a);//b=5intc=strlen(a);//c=3sizeof()是一种单目操作符,是用来计算你

    2022年4月8日
    34
  • 常见端口渗透总结[通俗易懂]

    常见端口渗透总结[通俗易懂]文章目录0x00背景服务默认端口爆破0x01实战测试文件共享服务端口渗透ftp服务NFS服务Samba服务LDAP协议远程连接服务端口渗透SSH服务Telnet服务Windows远程连接VNC服务Pcanywhere服务Web应用服务端口渗透IIS服务Apache/Tomcat/Nginx/Axis2WebLogicJbossWebsphereGlassFishJenkinsResinJettyLotus数据库服务端口渗透MySQL数据库MSSQL数据库Oracle数据库PostgreSQL数据库Mon

    2022年9月12日
    3

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号