阿里巴巴面试之利用两个int值实现读写锁
首先我们对读写锁做一个概述:
假设你的程序中涉及到对一些共享资源的读和写操作,且写操作没有读操作那么频繁。在没有写操作的时候,两个线程同时读一个资源没有任何问题,所以应该允许多个线程能在同时读取共享资源。但是如果有一个线程想去写这些共享资源,就不应该再有其它线程对该资源进行读或写,也就是说:读-读能共存,读-写不能共存,写-写不能共存。这就需要一个读/写锁来解决这个问题。
阿里巴巴四面最后的问题就涉及到了读写锁的实现。
问:对JAVA中的读写锁熟悉吗?
我:还可以...
问:简单介绍一下...
我:这是JAVA中的集合类,ReentrantReadWriteLock,分别处理读与写的操作,读读可以共存,读写不能共存,写写不能共存...
问:你自己实现一个读写锁
我:用synchronized,用一个队列...
问:不能那么麻烦,就用两个int值
我:两个int值,那一定是一个标记读次数,一个标记写次数啊,于是想了一会我想出了以下答案:


问:你这个可以是可以,但是如果读操作大量的话,写操作会产生饥饿等待...如何改进?
我想了想,是哦,然后脑子就短路了,挣扎,再挣扎...
我:给写加上优先级
问:不可以,就两个int
我:读次数设置一个阈值,超出主动释放
问:不可以,就两个int
我一脸懵逼...脑补画面
问:提醒你一下,lockwrite那里改一下
面试官发音不标准,我听了个lockread,想了半天,实在不知道怎么改,随便说了一个
问:我让你改lockwrite...
我:额,我看看...
又是漫长的等待...
我:没想出来...
问:把你那个条件拆开,让写占个坑...
我恍然大悟,我去...
然后他大概说了一遍,就草草结束了,郁闷中...
应该改成:

回来之后,大概看了一下,把整个程序写了一下:
package com.darrenchan.lock; /**
* 用两个int变量实现读写锁
* @author Think
*
*/
public class MyReadWriteLock { private int readcount = 0;
private int writecount = 0; public void lockread() throws InterruptedException{
while(writecount > 0){
synchronized(this){
wait();
}
}
readcount++;
//进行读取操作
System.out.println("读操作");
} public void unlockread(){
readcount--;
synchronized(this){
notifyAll();
}
} public void lockwrite() throws InterruptedException{
while(writecount > 0){
synchronized(this){
wait();
}
}
//之所以在这里先++,是先占一个坑,避免读操作太多,从而产生写的饥饿等待
writecount++;
while(readcount > 0){
synchronized(this){
wait();
}
}
//进行写入操作
System.out.println("写操作");
} public void unlockwrite(){
writecount--;
synchronized(this){
notifyAll();
}
} public static void main(String[] args) throws InterruptedException {
MyReadWriteLock readWriteLock = new MyReadWriteLock();
for(int i = 0; i < 2; i++){
Thread2 thread2 = new Thread2(i, readWriteLock);
thread2.start();
} for (int i = 0; i < 10; i++) {
Thread1 thread1 = new Thread1(i, readWriteLock);
thread1.start();
} } } class Thread1 extends Thread{
public int i;
public MyReadWriteLock readWriteLock; public Thread1(int i, MyReadWriteLock readWriteLock) {
this.i = i;
this.readWriteLock = readWriteLock;
} @Override
public void run() {
try {
readWriteLock.lockread();
Thread.sleep(1000);//模拟耗时
System.out.println("第"+i+"个读任务");
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
readWriteLock.unlockread();
}
}
} class Thread2 extends Thread{
public int i;
public MyReadWriteLock readWriteLock; public Thread2(int i, MyReadWriteLock readWriteLock) {
this.i = i;
this.readWriteLock = readWriteLock;
} @Override
public void run() {
try {
readWriteLock.lockwrite();
Thread.sleep(1000);
System.out.println("第"+i+"个写任务");
} catch (InterruptedException e) {
e.printStackTrace();
} finally {
readWriteLock.unlockwrite();
}
}
}
执行结果:
写操作
第0个写任务
读操作
读操作
读操作
读操作
读操作
读操作
读操作
读操作
读操作
读操作
第9个读任务
第3个读任务
第1个读任务
第8个读任务
第2个读任务
第0个读任务
第7个读任务
第4个读任务
第5个读任务
第6个读任务
写操作
第1个写任务
当然这是简单的,还可以进一步加深,可以参考博客:http://ifeve.com/read-write-locks/#simple
注:以上代码在++和--的时候仍然会产生并发异常,建议用AtomicInteger类型,在硬件上保证++和--操作不会出现并发异常。
阿里巴巴面试之利用两个int值实现读写锁的更多相关文章
- 用两个int值实现读写锁
private int readcount = 0; private int writecount = 0; public void lockread() throws InterruptedExce ...
- 两个int值相乘超过int最大值
System.out.println(100000000*1000 ); //输出结果是:1215752192 先将100000000*1000 的结果转化为二进制: 10111 01001000 0 ...
- 并发系列5-大白话聊聊Java并发面试问题之微服务注册中心的读写锁优化【石杉的架构笔记】
- 问题:两个对象值相同(x.equals(y) == true),但是可能存在hashCode不同吗?
面试官的考察点 这道题仍然是考察JVM层面的基本知识,面试官认为,基本功扎实,才能写出健壮性和稳定性很高的代码. 涉及到的技术知识 (x.equals(y)==true),这段代码,看起来非常简单,但 ...
- Flash图解线程池 | 阿里巴巴面试官希望问的线程池到底是什么?
前言 前几天小强去阿里巴巴面试Java岗,止步于二面. 他和我诉苦自己被虐的多惨多惨,特别是深挖线程和线程池的时候,居然被问到不知道如何作答. 对于他的遭遇,结合他过了一面的那个嘚瑟样,我深表同情(加 ...
- 牛客网Java刷题知识点之Map的两种取值方式keySet和entrySet、HashMap 、Hashtable、TreeMap、LinkedHashMap、ConcurrentHashMap 、WeakHashMap
不多说,直接上干货! 这篇我是从整体出发去写的. 牛客网Java刷题知识点之Java 集合框架的构成.集合框架中的迭代器Iterator.集合框架中的集合接口Collection(List和Set). ...
- two pointers思想 ---- 利用两个i, j两个下标,同时对序列进行扫描,以O(n)复杂度解决问题的一种思想
two pointers思想 ---- 利用两个i, j两个下标,同时对序列进行扫描,以O(n)复杂度解决问题的一种思想, 如果能用这种思想解决问题,那么会大大降低程序的复杂度. 两个利用这个思想的例 ...
- 阿里巴巴--java多线程的两种实现方式,以及二者的区别
阿里巴巴面试的时候,昨天问了我java面试的时候实现java多线程的两种方式,以及二者的区别当时只回答了实现线程的两种方式,但是没有回答上二者的区别: java实现多线程有两种方式: 1.继承Thre ...
- 利用checkbox的到值,并且存到数据库修改的话要显示之前选择的
在前台当然是利用checkbox来得到复选框的语言:{% for language in languages%}<input type="checkbox" name=&qu ...
随机推荐
- BZOJ.3329.Xorequ(数位DP)
题目链接 x^3x=2x -> x^2x=3x 因为a^b+((a&b)<<1)=a+b,x^2x=x+2x,所以x和2x的二进制表示中不存在相邻的1. (或者,因为x+2x ...
- 潭州课堂25班:Ph201805201 WEB 之 JS 第四课 (课堂笔记)
JS 引入方式 在 HTML 中写入 写在 的标签里 <script> </script>推荐 放在 </body> 结束之前 <!DOCTYPE html& ...
- 潭州课堂25班:Ph201805201 并发(通信) 第十三课 (课堂笔记)
from multiprocessing import Process # 有个 url 列表 ,有5个 url ,一次请求是1秒,5个5秒 # 要求1秒把 url 请求完, a = [] # 在进程 ...
- BZOJ2861 : 双向边定向为单向边
将每条双向边拆成两条单向边,若两条边中至少存在一条边使得删掉它之后图中SCC个数不变,则这条边可以定向. 将边中间加上点,变成删点问题. 对于每个SCC单独考虑,随便选择一个不是拆点出来的点S作为源. ...
- C语言学习中遇到的小问题(一)
C语言小白学习C语言的记录1 一.scanf一次性接收连续的数字 1.已知数量,且个数较少:scanf("%d%d%d",&a&b&c); 2.已知数量,但 ...
- ecshop 添加后台页面以及设置权限
转自 http://blog.csdn.net/tgh1981/article/details/10394059 ecshop 添加新页面 给ecshop后台增加管理功能页面 比如我们增加一个统计报表 ...
- Python和Mysql、Nginx
链接: python入门和基础: Python 中文学习大本营 你是如何自学 Python 的? 简明 Python 教程 给伸手党的福利:Python 新手入门引导 <Python爬虫学习系列 ...
- 读写锁--DEMO
package com.demo.read.write.lock; import java.util.HashMap; import java.util.Map; import java.util.c ...
- spring cloud: 升级到spring boot 2.x/Finchley.RELEASE遇到的坑
spring boot2.x已经出来好一阵了,而且spring cloud 的最新Release版本Finchley.RELEASE,默认集成的就是spring boot 2.x,这几天将一个旧项目尝 ...
- Drawing line on a click on ZedGraph Pane
https://stackoverflow.com/questions/12422398/drawing-line-on-a-click-on-zedgraph-pane public Form1() ...