网易一面,痛失30K:为啥用阻塞队列,list不行吗?

文章详细介绍了阻塞队列的概念,与List和普通Queue的区别,以及其在多线程环境中的作用和功能,强调了阻塞队列在生产者/消费者模式中的重要性,并提到了其核心方法如take和put。文章还提及了阻塞队列的有界和无界特性,并提供了相关面试题的解答。

说在前面

在40岁老架构师 尼恩的读者交流群(50+)中,最近有小伙伴拿到了一线互联网企业如网易、极兔、有赞、希音、百度、美团的面试资格,遇到一个很重要的面试题:

为啥要用阻塞队列,用list不行吗?

阻塞队列,是面试的绝对重点和难点。小伙伴没有答上来,痛失30K月薪。

这里尼恩给大家做一下系统化、体系化的线程池梳理,使得大家可以充分展示一下大家雄厚的 “技术肌肉”,让面试官爱到 “不能自已、口水直流”

也一并把这个题目以及参考答案,收入咱们的 《尼恩Java面试宝典》V89版本,供后面的小伙伴参考,提升大家的 3高 架构、设计、开发水平。

注:本文以 PDF 持续更新,最新尼恩 架构笔记、面试题 的PDF文件,请到公号【技术自由圈】获取

1、什么是阻塞队列?

阻塞队列是一种队列,阻塞队列是一种特殊的队列。

阻塞队列是一种可以在多线程环境下使用,并且支持阻塞等待的队列。

线程 1 往阻塞队列中添加元素,当阻塞队列是满的,线程 1就会阻塞,直到队列不满

线程 2 从阻塞队列中移除元素,当阻塞队列是空的,线程 2 会阻塞,直到队列不空;

2、主要并发队列关系图

上图展示了 Queue 最主要的实现类,可以看出 Java 提供的线程安全的队列(也称为并发队列)分为阻塞队列和非阻塞队列两大类。

BlockingQueue 下面有 6 种最主要的阻塞队列实现,分别是

  • ArrayBlockingQueue、
  • LinkedBlockingQueue、
  • SynchronousQueue、
  • DelayQueue、
  • PriorityBlockingQueue
  • LinkedTransferQueue。

非阻塞并发队列的典型例子是 ConcurrentLinkedQueue,这个类不会让线程阻塞,利用 CAS 保证了线程安全。

我们可以根据需要自由选取阻塞队列或者非阻塞队列来满足业务需求。

还有一个和 Queue 关系紧密的 Deque 接口,它继承了 Queue,如代码所示:

public interface Deque<E> extends Queue<E> {//...}

Deque 的意思是双端队列,音标是 [dek],是 double-ended-queue 的缩写,它从头和尾都能添加和删除元素;而普通的 Queue 只能从一端进入,另一端出去。这是 Deque 和 Queue 的不同之处,Deque 其他方面的性质都和 Queue 类似。

3、阻塞队列和 List、Set 的区别是什么?

阻塞队列和 List、Set 一样都继承自 Collection。

阻塞队列它和 List 的区别在于,List 可以在任意位置添加和删除元素。

而阻塞队列属于 Queue 队列的一种,Queue 只有两个操作:

  • 把元素添加到队列末尾;
  • 从队列头部取出元素。

超市的收银台就是一个队列:

我们常用的 LinkedList 就可以当队列使用,实现了 Dequeue 接口,还有 ConcurrentLinkedQueue,他们都属于非阻塞队列。

4、阻塞队列和普通Queue 队列的区别是什么?

阻塞队列和一般的队列的区别就在于:

  • 多线程环境支持,多个线程可以安全的访问队列
  • 支持生产和消费等待,多个线程之间互相配合,在某些情况下会挂起线程,一旦条件满足,被挂起的线程又会自动被唤醒
  • 当阻塞队列是空的,消费线程会阻塞,从队列中获取元素的操作将会被阻塞,直到队列不空
  • 当阻塞队列是满的,生产线程就会阻塞,往队列里添加元素的操作将会被阻塞,直到队列不满

5、阻塞队列的作用

阻塞队列,也就是 BlockingQueue,它是一个接口,如代码所示:

public interface BlockingQueue<E> extends Queue<E>{...}

BlockingQueue 继承了 Queue 接口,是队列的一种。

Queue 和 BlockingQueue 都是在 Java 5 中加入的。

BlockingQueue 是线程安全的,在很多场景下都可以利用线程安全的队列来优雅地解决业务自身的线程安全问题。

比如说,使用生产者/消费者模式的时候,生产者只需要往队列里添加元素,而消费者只需要从队列里取出它们就可以了,如图所示:

在图中,左侧有三个生产者线程,它会把生产出来的结果放到中间的阻塞队列中,而右侧的三个消费者也会从阻塞队列中取出它所需要的内容并进行处理。因为阻塞队列是线程安全的,所以生产者和消费者都可以是多线程的,不会发生线程安全问题。

既然队列本身是线程安全的,队列可以安全地从一个线程向另外一个线程传递数据,所以生产者/消费者直接使用线程安全的队列就可以,而不需要自己去考虑更多的线程安全问题。这也就意味着,考虑锁等线程安全问题的重任从“你”转移到了“队列”上,降低了开发的难度和工作量。

同时,队列还能起到一个隔离的作用

比如说开发一个银行转账的程序,那么生产者线程不需要关心具体的转账逻辑,只需要把转账任务,如账户和金额等信息放到队列中就可以,而不需要去关心银行这个类如何实现具体的转账业务。而作为银行这个类来讲,它会去从队列里取出来将要执行的具体的任务,再去通过自己的各种方法来完成本次转账。

这样就实现了具体任务与执行任务类之间的解耦,任务被放在了阻塞队列中,而负责放任务的线程是无法直接访问到银行具体实现转账操作的对象的,实现了隔离,提高了安全性。

6、阻塞队列的功能

阻塞队列区别于其他类型的队列的最主要的特点就是“阻塞”这两个字,

所以下面重点介绍阻塞功能:阻塞功能使得生产者和消费者两端的能力得以平衡,当有任何一端速度过快时,阻塞队列便会把过快的速度给降下来。

7、阻塞队列的核心方法

方法类型抛出异常特殊值阻塞超时
插入add(e)offer(e)put(e)offer(e,time,unit)
移除remove()poll()take()poll(time,unit)
检查elementpeek不可用不可用
  1. 抛异常的方法 就是在插入满了之后,会报一个异常,remove一样,element是检查队头的元素或者是否为空。
  2. 特殊值的方法是在插入满之后返回值变成了false而不是一个异常,取出失败的时候返回null。
  3. 阻塞方法是在插入满之后把这个方法阻塞,一直等待队列空出来一个之后再进行加入,会出现一直等待,也可能出现饥饿现象。
  4. 超时方法的话,当阻塞队列满时,队列会阻塞生产者线程一定时间,超过限时后生产者线程会退出。

实现阻塞最重要的两个方法是 take 方法和 put 方法。

7.1 take 方法

take 方法的功能是获取并移除队列的头结点,通常在队列里有数据的时候是可以正常移除的。

可是一旦执行 take 方法的时候,队列里无数据,则阻塞,直到队列里有数据。

一旦队列里有数据了,就会立刻解除阻塞状态,并且取到数据。

过程如图所示:

7.2 put 方法

put 方法插入元素时,如果队列没有满,那就和普通的插入一样是正常的插入,但是如果队列已满,那么就无法继续插入,则阻塞,直到队列里有了空闲空间。

如果后续队列有了空闲空间,比如消费者消费了一个元素,那么此时队列就会解除阻塞状态,并把需要添加的数据添加到队列中。

put 过程如图所示:

以上过程中的阻塞和解除阻塞,都是 BlockingQueue 完成的,不需要我们自己处理。

7.3 是否有界(容量有多大)

此外,阻塞队列还有一个非常重要的属性,那就是容量的大小,分为有界和无界两种。

  • 有的阻塞队列是无界的,无界队列意味着里面可以容纳非常多的元素,例如 LinkedBlockingQueue 的上限是 Integer.MAX_VALUE,约为 2 的 31 次方,是非常大的一个数,可以近似认为是无限容量,因为几乎无法把这个容量装满。
  • 但是有的阻塞队列是有界的,例如 ArrayBlockingQueue 如果容量满了,也不会扩容,所以一旦满了就无法再往里放数据了。

尼恩提示

更多的 JUC 高并发知识,请参见《Java 高并发核心编程 卷2 加强版

说在后面

问题回答到这里,已经20分钟过去了,

既然 对阻塞队列 表达得那么娴熟,一定是遇到过很多的高并发场景,解决很多高并发问题,那么一定是技术大佬、技术高手,面试官已经爱到 “不能自已、口水直流” 啦。

offer,当然也就来了。

推荐相关阅读

滴滴太狠:分布式ID,如何达到1000Wqps?

10亿级用户,如何做 熔断降级架构?微信和hystrix的架构对比

虾皮一面:手写一个Strategy模式(策略模式)

腾讯太狠:40亿QQ号,给1G内存,怎么去重?

吃透8图1模板,人人可以做架构

《尼恩 架构笔记》《尼恩高并发三部曲》《尼恩Java面试宝典》PDF,请到下面公号【技术自由圈】取↓↓↓

<think>嗯,用户再次聚焦在立创EDA绘制STM32F103C8T6最小开发板的具体操作上,看来ta正在动手实践阶段。之前给的系统级资料概览可能够实操,这次需要拆解成真正的步骤指南。 用户引用的两个资料很关键:引用[1]提到借鉴江协科技的模块设计,引用[2]则具体到封装选择操作。这提示我应该把B站案例作为入口,同时强化封装选择的细节——毕竟LQFP48封装选错尺寸会导致生产事故。 注意到用户上次对"必可少组件"的追问,这次教程必须突出电源/复位/晶振这三个核心模块的设计要点。特别是BOOT0/BOOT1电路容易被新手忽略,但又是烧录的关键,得单独强调。 卡顿问题上次反馈较多,这次要在关键节点插入保存建议:比如原理图每完成1/3保存、放置高危元件(如STM32主芯片)后立刻保存。立创EDA的自动保存间隔是10分钟,手动存盘能避免痛失进度。 用户应该需要即时可用的封装参数,所以直接列出关键尺寸:主芯片7x7mm,滤波电容0805封装,晶振HC-49S。这些数值若让用户自己查会拖慢进度。 最后考虑加入"避坑指南":比如VREF+必须接滤波电容否则ADC准,USB-DP要串电阻防干扰。这些经验之谈能帮ta少走弯路。 </think>以下是使用立创EDA绘制STM32F103C8T6最小开发板的详细教程,结合引用资料和实际操作要点: --- ### **一、绘制步骤详解** 1. **创建工程与选择封装** - 打开立创EDA,新建工程并命名(如`STM32F103C8T6_MinSystem`) - **搜索器件**:左侧元件库输入 `STM32F103C8T6` → 选择 **LQFP_48封装(尺寸L7.0×W7.0)** [^2] - 放置主芯片:点击元件库搜索结果中的“放置”,拖动到画布中央 2. **添加核心电路模块**(参考最小系统设计[^1]) - **电源电路** - 放置 `0.1μF` 和 `10μF` 电容:分别连接 `VDD`(3.3V)和 `GND` - 电源滤波:每个 `VDD` 引脚就近添加 `0.1μF` 电容 - **复位电路** - 放置 `10kΩ` 电阻和 `100nF` 电容:串联在 `NRST` 和 `GND` 之间 - 添加复位按钮:并联在 `NRST` 和 `3.3V` 之间 - **晶振电路** - 放置 **8MHz** 晶振:连接 `OSC_IN`(PD0)和 `OSC_OUT`(PD1) - 添加负载电容:两个 `22pF` 电容分别接晶振引脚到 `GND` - **启动模式选择** - `BOOT0` 引脚:通过 `10kΩ` 电阻下拉到 `GND`(默认Flash启动) - 预留测试点:在 `BOOT0` 和 `3.3V` 间预留跳线帽接口 3. **关键布线技巧** - **电源网络**: - 使用 **铺铜** (快捷键 `P`)创建 `GND` 平面 - `3.3V` 走线宽度 ≥ 0.3mm(载流能力要求) - **信号线优化**: - 高速信号(如 `SWD` 调试口)走线尽量短直 - 避免 `晶振线路` 与高频信号平行 4. **PCB布局规范** ```markdown | 模块 | 布局要求 | |--------------|-----------------------------| | 电源滤波电容 | 靠近MCU的VDD引脚(<3mm) | | 晶振 | 远离干扰源(如电源、USB接口)| | 复位电路 | 靠近NRST引脚 | | SWD调试接口 | 放置在PCB边缘便于连接 | ``` --- ### **二、常见问题解决(引用[^3]扩展)** 1. **软件卡顿** - **对策**: - 关闭必要的图层(如丝印层) - 分模块绘制(先完成电源→复位→晶振→外设) - 每完成一个模块 **Ctrl+S保存** - ⚠️ 避免错误:勿在未保存时切换页面或版本 2. **封装错误** - **典型问题**:芯片引脚与焊盘匹配 - **验证方法**: 1. 双击元件→查看属性中的封装型号 2. 对照数据手册核对引脚间距(LQFP48引脚间距 **0.5mm**) 3. **电气规则检查(ERC)报错** - **高频错误**: - 未连接的电源网络 → 确认所有 `VDD/GND` 已标网络标签 - 引脚悬空 → 检查 `BOOT1` 等未用引脚需下拉处理 --- ### **三、设计验证与生产** 1. **DRC检测**: - 设置规则:线距 ≥ 0.2mm,线宽 ≥ 0.25mm - 重点检查:晶振区域禁止其他信号层穿越 2. **生成制造文件**: - 顶部菜单 → 文件 → 导出 → **Gerber文件** - 直接使用立创 **PCB下单** 服务(支持Gerber自动解析) --- ### **四、参考文献与资源** - [^1] 最小系统参考设计:江协科技B站模块(精简电路,适合入门) - [^2] 封装选择指南:LQFP_48封装操作流程 - [^3] 社区避坑:立创EDA常见故障解决方案库 ---
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值