Java线程学习-Lock

Lock

Lock是Java提供给开发者JDK级别的一种控制代码同步访问的工具类。只有一个子类ReentrantLock。功能和synchronized差不多。使用也是非常的简单

class Test {
    private Lock lock = new ReentrantLock();
  
    //需要同步访问的方法
    private void doSomething() {
        lock.lock();
        
        try {
            //do something
        } catch (Exception e) {

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

例子

    private void lockTest() {
        sendMessage("lockTest begin");
        new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 10; i++) {
                    doSomething();
                }
                sendMessage("thread-1 end");
            }
        }, "thread-1").start();
        new Thread(new Runnable() {
            @Override
            public void run() {
                for (int i = 0; i < 10; i++) {
                    doSomething();
                }
                sendMessage("thread-2 end");
            }
        }, "thread-2").start();
    }

   //一次访问这个函数
    private void doSomething() {
        lock.lock();

        try {
            Thread.sleep(100);
            sendMessage(Thread.currentThread().getName() + " doSomething");
        } catch (InterruptedException e) {
            e.printStackTrace();
        }finally {
            lock.unlock();
        }
    }

输出:


image.png

ReadWriteLock

同步读写操作的一个锁。和Lock完全没有任何关系。

  • 允许多个线程同时进行读操作。
  • 不允许多个线程同时写操作,只会谁拿到锁谁先去执行写操作。
  • 多个线程既有读又有写操作,没有线程进入写或者等待写,就可以多个线程读了;进入写操作后,不允许其他线程读或者写
读操作
    private void readLockTest() {
        new Thread(new Runnable() {
            @Override
            public void run() {
                readSomething();
            }
        }, "t1").start();

        new Thread(new Runnable() {
            @Override
            public void run() {
                readSomething();
            }
        }, "t2").start();
    }

    private void readSomething() {
        readWriteLock.readLock().lock();
        try {
            Iterator<String> iterator = cacheList.iterator();
            while (iterator.hasNext()) {
                Thread.sleep(100);
                sendMessage(Thread.currentThread().getName() + "读取 -- "+ iterator.next());
            }
        }catch (InterruptedException e) {
            e.printStackTrace();
        }finally {
            readWriteLock.readLock().unlock();
        }
    }

输出:


image.png
写操作
   private void writeLockTest() {
        final Thread t1 = new Thread(new Runnable() {
            @Override
            public void run() {
                writeSomething();
            }
        }, "t1");
        t1.start();

        final Thread t2 = new Thread(new Runnable() {
            @Override
            public void run() {
                writeSomething();
            }
        }, "t2");
        t2.start();

        new Thread(new Runnable() {
            @Override
            public void run() {
                try {
                    t1.join();
                    t2.join();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                sendMessage("写完了");
                for (String s : cacheList) {
                    sendMessage(s);
                }
            }
        }).start();
    }

    private void writeSomething() {
        readWriteLock.writeLock().lock();
        try{
            for (int i = 0; i < 10; i++) {
                Thread.sleep(100);
                cacheList.add("这是第" + i + "元素");
                sendMessage(Thread.currentThread().getName() + "写入---这是第" + i + "元素");
            }
        }catch (InterruptedException e) {
            e.printStackTrace();
        }finally {
            readWriteLock.writeLock().unlock();
        }
    }

输出:


image.png
读写同时进行
    private void readWriteLockTest(){
        new Thread(new Runnable() {
            @Override
            public void run() {
                writeSomething();
            }
        }, "t1").start();

        new Thread(new Runnable() {
            @Override
            public void run() {
                readSomething();
            }
        }, "t2").start();

        new Thread(new Runnable() {
            @Override
            public void run() {
                readSomething();
            }
        }, "t3").start();
        new Thread(new Runnable() {
            @Override
            public void run() {
                writeSomething();
            }
        }, "t4").start();

输出:


image.png

测试方法中有四个线程,分别是:;日志中在t2读的时候,这是t3启动了,t3是读的操作,此时并没有写或者等待写的操作,所以t3也开始读,最终t2和t3同时读。

把测试方法顺序改一下,分别是:
输出:

image.png

可以看出在t2的时候,t3启动了,因为t3是写操作,所以t4启动后,因为t3已经等待写,所以t4会在t3写完后才执行。

demo源码

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 文章来源:http://www.54tianzhisheng.cn/2017/06/04/Java-Thread/...
    beneke阅读 5,639评论 0 1
  • 进程:正在执行的程序,是一个动态的过程 线程:是进程中用于控制程序执行的控制单元(执行路径,执行情景) 进程中至少...
    宝塔山上的猫阅读 3,269评论 0 1
  • 下面是我自己收集整理的Java线程相关的面试题,可以用它来好好准备面试。 参考文档:-《Java核心技术 卷一》-...
    阿呆变Geek阅读 15,041评论 14 507
  • *每一个生命都值得被珍惜。 2017年6月24日。 我的大学生活完完整整地结束了。 留下很多遗憾,留下很多无法弥补...
    新儿爷阅读 1,792评论 0 0
  • 目录 1、美好的愿望 2010年夏天,蓝心微16岁生日时许了个美好的愿望,记在了她心爱的粉色笔记本上:“我希望和萱...
    陆Nene阅读 3,114评论 2 2

友情链接更多精彩内容