自己动手写把”锁”之---锁的作用

前序
这是一个系列文章,前边几篇比较基础,主要为了后续做准备。熟悉的朋友可以直接跳过看后续的文章。
本主题很重要,学完这个系列,你将会对Java并包有一个透彻的原理性的认识。线程池技术、阻塞队列、信号量、原子性操作等等所用的基础技术都会在这系列的文章中讲到。学完之后,大家可以自己实现一个CountDownLatch来练练手。
 
正文
都知道,现在处理器的核数越来越多,为充分利用其计算资源,服务端编程通常会用上多线程技术。利用多线程技术可以同时进行计算任务,从而提高的服务的并发度。
 
但是,当多线程对同一块内存资源进行操作时,如果不对线程进行“排队”,其结果将是混乱不堪预测额。
 
这里说的“排队”就是通常说的线程同步。当一个线程在操作这个资源时,后来的线程等待上一个线程操作完才能开始。
 
举个代码栗子:
复制代码
public class TestMyLock {
    private static  List<Integer> list = new ArrayList<>();
    public static void main(String[] args){
        Thread t1 = new Thread(new Runnable() {
            @Override
            public void run() {
                for(int i=0;i<10000;i++){
                    add(i);
                }
            }
        });
        Thread t2 = new Thread(new Runnable() {
            @Override
            public void run() {
                print();
            }
        });
        t1.start();
        t2.start();
    }
    private static void add(int i){
        list.add(i);
    }
    private static void print(){
        Iterator<Integer> iterator = list.iterator();
        while (iterator.hasNext()){
            System.out.println(iterator.next());
        }
    }
}
复制代码

以上我们创建了两个线程:t1和t2。t1循环一万次向list里边添加数字,t2遍历list并将内容打印到控制台。

执行后得到如下错误:
 

 

原因很简单,就是两个线程同时操作list而没有进行线程同步导致的报错。我们修改下代码再来看一下。
add和print方法修改为如下:
复制代码
private synchronized static void add(int i){
        list.add(i);
}
private synchronized static void print(){
        Iterator<Integer> iterator = list.iterator();
        while (iterator.hasNext()){
            System.out.println(iterator.next());
        }
}
复制代码

我们仅仅是在add和print方法前加入了synchronized修饰词,程序便可以正常执行了。

 
synchronized关键字是Java内置的同步锁。两个线程会竞争对synchronized绑定的同步对象加锁,加锁失败的线程会阻塞等待加锁成功的线程执行完毕。由于add和print方法都是静态方法,这里synchronized绑定的同步对象就是TestMyLock.class。
 
----------------------------------------------------------------------------
 

我正在「JAVA互联网技术」和朋友们讨论有趣的话题,你一起来吧?

https://t.zsxq.com/EUn6IIE

 
 

posted @   清泉^_^  阅读(1217)  评论(1编辑  收藏  举报
编辑推荐:
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
阅读排行:
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· 阿里巴巴 QwQ-32B真的超越了 DeepSeek R-1吗?
· 【译】Visual Studio 中新的强大生产力特性
· 【设计模式】告别冗长if-else语句:使用策略模式优化代码结构
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
点击右上角即可分享
微信分享提示