跳转到内容

Blog

接触 Vibe Coding 八个多月后的感受

接触 Vibe Coding 已经八个多月了。

回头看,这段时间给我带来的变化非常大。它不只是让我多认识了一些 AI 工具,也不只是让我写代码的速度变快了。更重要的是,它改变了我看待开发这件事的方式。

以前我更关注技术本身。

我会想这个功能应该怎么实现,代码怎么写,框架怎么选,接口怎么设计,数据库结构怎么拆。很多时候,注意力会自然落在“如何把代码写出来”这件事上。

而现在,我越来越多地开始关注产品本身。

这个功能为什么要做?用户会怎么使用?流程是不是顺?页面是不是清楚?这个需求背后真正要解决的业务问题是什么?这些问题慢慢变得比“代码怎么写”更靠前。

这就是 Vibe Coding 对我最大的影响。

在更早的时候,我使用 AI 的方式其实很简单。

去年的上半年,AI 对我来说更多还是一个开发辅助工具。它主要停留在 Web 式的 Chat 形态里,我会问它一些问题,让它帮我解释概念、检索资料、分析报错、生成一些代码片段。

那个阶段的 AI 很像一个随时在线的助手。

它能帮我查东西,也能帮我补充思路,但大多数时候,真正的开发过程还是由我自己主导。我要自己拆任务、自己打开项目、自己修改文件、自己调试和验证。

AI 参与了过程,但没有真正进入开发工作流的中心。

那时候我对它的理解也很朴素:它可以提高效率,可以减少搜索成本,可以帮我更快理解一些不熟悉的知识。

但我还没有意识到,它会在后面改变整个开发方式。

后来,Agent 的概念越来越火。

我开始看到各种 CLI 工具出现,也开始频繁听到 token、上下文、模型网关、提示词、工具调用、代码代理这些词。AI 不再只是一个聊天窗口,它开始进入终端、进入编辑器、进入项目目录,甚至可以直接阅读代码、修改文件、运行命令、检查结果。

这和以前完全不一样。

以前是我把问题复制给 AI,然后把答案再搬回项目里。现在则更像是 AI 直接坐进了项目现场,和我一起看代码、改代码、验证代码。

这时我也开始加入 Vibe Coding 的行列。

刚开始的时候,我其实并不知道这些工具应该怎么用。面对各种模型、API、CLI、代理配置,我会有点茫然。它们看起来都很强,但真正落到自己的项目里,还是需要一段适应过程。

我需要理解它们的边界。

哪些事情可以交给它?哪些事情必须自己判断?什么时候应该让它改代码?什么时候只是让它分析?上下文应该怎么给?任务应该怎么拆?

这些并不是看一篇教程就能立刻掌握的。

后来我逐渐知道了 New API 这类整合型网关,也开始理解它们在 AI 工作流中的意义。

模型越来越多,不同模型有不同能力、价格和使用限制。如果每一个工具都单独配置,就会很分散。整合型网关的意义在于,它能把不同模型入口统一起来,让工具调用变得更稳定,也更容易管理。

再后来,我接触到了 Claude Code 这类工具。

它让我真正感受到“AI 参与编码”这件事和普通问答的区别。

普通问答更像是你问一句,它答一句。CLI 编码工具则更像是你把它放进项目里,它可以沿着任务往前走:阅读文件、理解结构、修改代码、运行检查、再根据结果继续调整。

这时候,AI 就不只是回答问题,而是在参与完成工作。

当然,这并不意味着我可以完全放手。

相反,我越来越感觉到,使用这类工具时,人要承担更高层次的判断。你要知道目标是什么,知道验收标准是什么,知道哪里不能乱动,知道生成的代码是否符合项目长期维护的方向。

AI 可以很快,但方向仍然要由人来定。

过去我做一个东西,第一反应常常是技术问题。

页面怎么写?接口怎么接?状态怎么管理?样式怎么调?

现在我会先想产品问题。

这个页面存在的目的是什么?用户第一眼应该看到什么?如果他想继续阅读,路径是不是顺?如果他在移动端打开,会不会困惑?如果内容越来越多,列表是否还能承载?如果未来要部署、维护、持续写文章,流程是不是足够轻?

这种变化很明显。

因为当 AI 可以承担大量具体编码工作后,我的注意力就被释放出来了。我不再需要把所有精力都压在每一行代码上,而是可以站得稍微高一点,看整个产品的结构和体验。

这并不是说技术不重要。

技术仍然重要,而且越到后面越重要。只是技术不再是唯一中心。它更像是实现产品目标的手段,而不是最终目的。

以前我可能会因为某个技术点很有意思就想做点东西。现在我会先问:这个东西解决了什么问题?它对用户、对内容、对长期维护有什么价值?

Vibe Coding 也让我更关注业务流程。

一个功能不是孤立存在的。它前面有入口,后面有结果,中间有状态变化和用户决策。只把某个页面写出来,并不代表功能真的完成。

比如一个博客站,不只是能展示文章就够了。

还要考虑:

  • 新文章怎么创建。
  • 分类和标签怎么维护。
  • 首页如何呈现内容价值。
  • 列表页如何让读者快速判断是否要点进去。
  • 文章页如何让阅读体验稳定。
  • 部署后如何只专注维护内容。

这些都不是单纯的代码问题,而是产品流程问题。

当 AI 能帮我更快完成具体实现后,我反而会花更多时间思考这些流程是否合理。

我会更在意一个功能放在系统里是不是自然,一个页面是不是为后续内容增长留好了空间,一个交互是不是符合读者直觉。

这其实是更接近产品视角的思考。

这八个多月里,我最大的感受是:AI 把很多“执行层面的阻力”变小了。

以前想到一个功能,可能要先考虑技术栈、查文档、写样板代码、调样式、修报错。很多时候,还没真正验证想法,就已经被实现细节消耗掉了。

现在不同了。

我可以更快把想法变成可运行的东西,再通过实际效果判断它是否值得继续优化。

这会让开发节奏发生变化。

以前更像是先想很久,再动手实现。现在更像是先做出一个版本,然后不断观察、调整、迭代。

AI 给我的不是简单的偷懒,而是更短的反馈周期。

当反馈周期变短,人就更容易围绕结果做判断,而不是长时间停留在假设里。

使用 Vibe Coding 越久,我越觉得人并没有变得不重要。

相反,人变得更重要。

因为 AI 可以生成代码,但它不一定知道什么是适合你的。它可以给出方案,但它不知道你真正想要的产品气质。它可以完成任务,但它不会天然理解你的长期规划。

所以,人需要做这些事情:

  • 定义目标。
  • 拆解任务。
  • 判断取舍。
  • 控制范围。
  • 验收结果。
  • 维护产品方向。

如果没有这些判断,AI 很容易把事情做得很快,但不一定做得正确。

这也是我慢慢学到的一点:Vibe Coding 不是随便让 AI 写代码,而是学会用清晰的目标和上下文引导它,把人的判断和 AI 的执行力结合起来。

这段经历让我发生了几个明显变化。

第一,我更愿意从产品角度看问题。

我不再只问“这个功能怎么写”,而是会先问“这个功能为什么存在”。

第二,我更重视流程。

页面、内容、工具、部署、维护,它们应该连成一条顺畅的链路,而不是一个个孤立的点。

第三,我对学习 AI 相关知识更有动力。

从模型到上下文,从 CLI 工具到网关,从提示词到任务拆解,这些内容不再只是概念,而是会真实影响我每天的开发方式。

第四,我开始更相信个人项目的可能性。

以前一个人做完整产品会觉得很重。现在虽然仍然不轻松,但至少很多原本消耗人的细节可以被 AI 分担。一个人的上限,正在被工具重新拉高。

接触 Vibe Coding 的这八个多月,对我来说像是一次开发方式的迁移。

我从把 AI 当作问答工具,慢慢转向把它当作开发协作者。也从更关注技术实现,逐渐转向关注产品本身、业务流程和最终结果。

这种变化不是一夜之间发生的。

它是在一次次尝试工具、配置模型、修改项目、验证结果的过程中慢慢形成的。

现在的我依然还在学习 AI,也还在摸索更适合自己的工作流。但有一点已经很明确:未来的开发不会再回到过去那种完全依赖手工推进的状态。

AI 会继续进入开发流程,而我需要做的,是学会站在更高的位置使用它。

把注意力从代码细节里适当抽出来,更多地放到产品、体验、流程和价值上。

这可能就是 Vibe Coding 最吸引我的地方。

它不是让我不再关心技术,而是让我终于有更多精力去关心技术背后真正要完成的事情。

梦开始的地方

这是我创建项目的第一篇博客文章。

在这个博客里,我将分享我的技术学习历程、项目经验以及一些随笔思考。希望通过这个平台,能够记录下我的成长轨迹,也能与志同道合的朋友们交流和分享。

我的名字叫做 veyliss,这是我“梦开始的地方”。我希望在这里能够记录下我的梦想和努力的过程,也希望能够激励自己不断前行。无论是技术上的突破,还是生活中的点滴,我都希望能够在这里留下足迹。

在许多年前我就曾想着自己搭建真正属于自己的博客网站,记录自己的学习和成长。如今这个想法终于实现了,我感到非常兴奋和满足。这个博客不仅是一个记录工具,更是一个激励自己不断前进的动力源泉。

我相信,通过这个博客,我能够更好地总结和反思自己的学习过程,也能够与更多的人分享我的经验和见解。无论是技术上的问题,还是生活中的思考,我都希望能够在这里找到共鸣和支持。

最后,感谢每一个来到这个博客的朋友们,希望你们能够在这里找到有价值的内容,也希望我们能够在这里一起成长和进步。让我们一起在这个梦开始的地方,书写属于我们的故事吧!

回顾 2025

回望 2025 年,时间像被按下了快进键。

这一年走得很快,也走得并不轻松。很多事情在开始时都带着热情,真正走到最后却发现,能完整收尾的并不算多。想做的项目、想沉淀的知识、想持续推进的计划,有些还停留在半成品阶段,有些甚至只是短暂地闪过念头。

这并不是一个让人完全满意的年份。

但它也不是毫无意义的一年。

这一年里,有一段连续而紧张的工作经历。

那几个月很充实,也很消耗人。每天都在处理新的任务,面对新的问题,试着把一些看起来并不容易的事情往前推进。工作中,我尽力认真对待每一项安排,也不断尝试挑战自己原本觉得困难的部分。

这段经历让我收获了不少东西。

我积累了更真实的工作经验,也接触到了很多优秀的同事和前辈。从他们身上,我看到了更成熟的处理方式、更稳定的职业节奏,以及面对复杂问题时更清晰的判断。

这些东西不是单靠看教程、写练习就能得到的。它们来自具体的工作现场,来自一次次沟通、交付、修改和复盘。

不过遗憾也很明显。

个人能力确实在提升,但还没有达到自己期待中的飞跃。和行业里真正优秀的人相比,差距依然存在,而且并不小。意识到这一点的时候,会有一点失落,但也会更清楚自己接下来应该往哪里用力。

下班后的疲惫,是这一年很真实的一部分。

这种疲惫不只是身体上的,更是精神上的。忙碌一天之后,大脑长时间处在紧绷状态。回到住处,整个人像被抽空了能量,明明知道还有很多东西值得学习,却很难再把自己重新拉回专注状态。

我曾经计划利用业余时间继续学习新知识,补足能力短板,也想持续推进一些个人项目。可是很多时候,打开资料、看到待办列表,就会先感到一阵无力。

于是一些计划被推迟,一些想法被搁置,一些原本应该继续打磨的东西,也慢慢停在了半路。

这并不值得美化。

它只是提醒我:人的精力是有限的,自我提升也不能只靠一时热情。真正能走得远的,应该是更稳定、更可持续的节奏。

这一年,我也多次想过要构建自己的知识体系。

我越来越能感受到,零散学习带来的问题很明显。今天看一点后端,明天补一点前端,后天又去了解新的工具和概念,短期内似乎学了很多,但如果没有整理和连接,这些知识很容易散落在各处。

我希望能把这些零散的知识点串联起来。

不是为了把自己包装得很厉害,而是希望在未来遇到问题时,能更快找到方向,知道某个知识点在整个技术体系里处于什么位置,也能把过去踩过的坑、做过的项目、解决过的问题留下来。

这也是我后来越来越想认真维护博客和知识库的原因。

文字不是为了证明什么,而是为了给自己留下路径。现在的记录,也许会成为未来某个阶段重新出发时的坐标。

2025 年,我结识了很多圈内朋友。

他们来自不同领域,有交易所、Web3、外卖行业、传统行业、SaaS 领域,也有来自大厂的朋友。还有一些自由职业者,他们的工作方式和生活状态让我产生过很多羡慕与向往。

和这些人交流,会明显感觉到世界比自己日常接触到的范围更大。

不同的人在不同赛道里寻找机会,也在用各自的方式处理工作、生活和成长之间的关系。有的人很专注,有的人很灵活,有的人已经找到了相对自由的节奏,也有人还在变化里持续摸索。

这些交流让我意识到,职业发展不是只有一条路。

稳定工作是一种选择,持续深耕是一种选择,做项目、做产品、做自由职业,也都是不同的选择。重要的是,自己要逐渐知道想过怎样的生活,并为之积累足够的能力。

2025 年也是 AI 快速爆发的一年。

越来越多行业开始投入 AI 相关研发,尝试用 AI 提升效率、优化流程、创造新的业务价值。无论是技术研发、内容生产,还是产品设计、数据分析,AI 都在逐渐进入日常工作。

这不是一个遥远的趋势,而是正在发生的变化。

我能明显感觉到,未来互联网生态会和 AI 更紧密地连接在一起。很多岗位的工作方式会被改变,很多工具会被重做,很多原本依赖人工经验的流程,也会被新的方式重新组织。

这对个人来说既是压力,也是机会。

压力在于,原本掌握的技能可能很快变得不够用。机会在于,如果能更早理解这些变化,并把 AI 当成能力放大器,就有可能在新的阶段找到更好的位置。

2025 年并不是一个完成度很高的年份。

它有遗憾,有拖延,有没有收尾的项目,也有很多没有真正落实的计划。但它同样留下了工作经验、人际连接、行业观察和对自我节奏的重新认识。

我不想把这一年写得过于漂亮。

因为真实的成长并不总是热血的。很多时候,它只是一次次发现自己的不足,然后在疲惫里慢慢调整方向。

如果要给 2025 年一个总结,我想它更像是一个提醒:

不要只依赖热情,也不要害怕缓慢。

真正重要的,是在每一次停顿之后,还能重新开始。

Java 校招面试题复盘清单

这是一份 Java 校招面试题复盘清单。

它不适合当作“背诵稿”逐字记忆,更适合当作复习地图:先知道面试会问哪些方向,再把每个问题整理成可以讲清楚的核心答案。

Java 校招面试通常会围绕三部分展开:

模块占比重点
技术基础约 40%Java、Spring、MySQL、Redis、JVM、并发
项目深挖约 40%项目背景、技术选型、难点、优化、问题排查
学习能力约 20%最近在学什么、为什么做 Java、如何解决问题

如果项目里有 Elasticsearch,那么 ES 往往会成为面试官重点追问的方向。

HashMap 底层主要是数组、链表和红黑树。

当放入一个 key-value 时,会先根据 key 的 hashCode() 计算 hash,再定位到数组下标。如果该位置没有元素,就直接放入;如果已经有元素,就会形成链表或红黑树。

在 Java 8 之后,当链表长度达到一定阈值,并且数组容量足够大时,链表会转换为红黑树,用来提高查询效率。

可以这样回答:

HashMap 通过 hash 定位数组下标,数组中每个位置叫 bucket。发生 hash 冲突时,Java 8 以前主要用链表,Java 8 之后链表过长会转为红黑树。扩容时会重新计算元素位置,所以 HashMap 的性能和初始容量、负载因子、hash 分布都有关系。

HashMap 没有做同步控制,多线程同时读写时可能出现数据覆盖、状态不一致、扩容异常等问题。

常见风险:

  • 多个线程同时 put,可能覆盖彼此写入。
  • 扩容时结构变化,其他线程同时访问可能拿到异常结果。
  • 统计数量 size 可能不准确。

所以多线程场景通常使用 ConcurrentHashMap

ConcurrentHashMap 如何实现线程安全

Section titled “ConcurrentHashMap 如何实现线程安全”

Java 8 中,ConcurrentHashMap 主要通过 CAS 和 synchronized 保证线程安全。

它不是给整张表加一把大锁,而是尽量缩小锁粒度。

常见回答:

Java 8 的 ConcurrentHashMap 底层也是数组、链表和红黑树。插入时,如果桶为空,会通过 CAS 放入节点;如果桶不为空,会对桶头节点加 synchronized,只锁当前桶。这样既保证线程安全,又比 Hashtable 整表加锁性能更好。

ArrayList 底层是动态数组,LinkedList 底层是双向链表。

对比项ArrayListLinkedList
底层结构动态数组双向链表
随机访问快,按下标访问 O(1)慢,需要遍历
中间插入删除需要移动元素找到节点后修改指针
内存占用相对较少每个节点要存前后指针
常用场景查询多插入删除多,但实际也要看位置

面试里要注意:不要简单说 LinkedList 插入删除一定快。因为如果要先按索引找到位置,遍历本身也有成本。

String、StringBuilder、StringBuffer 的区别

Section titled “String、StringBuilder、StringBuffer 的区别”

String 是不可变对象,每次修改都会产生新字符串。

StringBuilder 是可变字符序列,线程不安全,但性能较好。

StringBuffer 也是可变字符序列,方法加了同步,线程安全,但性能通常低于 StringBuilder

常见选择:

  • 少量字符串拼接:直接用 String
  • 单线程大量拼接:用 StringBuilder
  • 多线程共享拼接对象:用 StringBuffer,但实际业务中较少这样用。

== 比较的是两边是否相等。

对于基本类型,比较的是值。

对于引用类型,比较的是对象地址。

equals() 是对象方法,默认实现也是比较地址,但很多类会重写它,比如 String 会比较字符串内容。

equals-demo.java
String a = new String("hello");
String b = new String("hello");
System.out.println(a == b); // false
System.out.println(a.equals(b)); // true

Java 异常体系的顶层是 Throwable

它下面主要分为:

  • Error:严重错误,程序通常不主动处理,比如 OutOfMemoryError
  • Exception:程序可以捕获和处理的异常。

Exception 又分为:

  • checked exception:编译期异常,必须处理或声明抛出。
  • unchecked exception:运行时异常,继承自 RuntimeException,例如空指针、数组越界。

面试里可以补一句:业务开发中不要滥用异常控制正常流程,异常更适合表示非预期情况。

反射是 Java 在运行时获取类信息、创建对象、调用方法、访问字段的能力。

常见应用场景:

  • Spring 创建 Bean、依赖注入。
  • MyBatis 映射对象字段。
  • 注解解析。
  • 测试框架调用测试方法。
  • 动态代理。

反射灵活,但也有缺点:性能相对普通调用更低,可读性和安全性更差。

进程是操作系统资源分配的基本单位。一个应用程序运行起来通常就是一个进程。

线程是 CPU 调度的基本单位,一个进程中可以包含多个线程。

可以这样回答:

进程拥有独立的内存空间,线程共享同一进程的内存资源。线程切换成本通常低于进程,但共享数据也会带来线程安全问题。

synchronized 可以修饰方法或代码块,用来保证同一时间只有一个线程进入临界区。

它依赖对象监视器锁,也就是 monitor。

进入同步代码块时,线程尝试获取对象锁;执行完或异常退出时释放锁。

可以补充:

  • 修饰普通方法,锁的是当前对象 this
  • 修饰静态方法,锁的是当前类的 Class 对象。
  • 修饰代码块,可以指定锁对象。

volatile 主要有两个作用:

  • 保证变量对多线程的可见性。
  • 禁止指令重排序。

它不能保证复合操作的原子性。

比如 count++ 包含读取、加一、写回,不是一个原子操作,所以只加 volatile 仍然不安全。

常见场景:

  • 状态标志位。
  • 单例模式双重检查锁中的实例变量。

线程池是提前创建并管理一组线程,任务来了以后交给线程池执行。

使用线程池的原因:

  • 避免频繁创建和销毁线程。
  • 控制并发线程数量。
  • 提高响应速度。
  • 统一管理任务队列、拒绝策略和线程生命周期。

ThreadPoolExecutor 常见核心参数:

参数作用
corePoolSize核心线程数
maximumPoolSize最大线程数
keepAliveTime非核心线程空闲存活时间
unit时间单位
workQueue任务队列
threadFactory线程创建工厂
handler拒绝策略

执行流程可以概括为:

核心线程未满 -> 创建核心线程
核心线程已满 -> 放入任务队列
队列满了 -> 创建非核心线程
线程数达到最大且队列也满 -> 执行拒绝策略

死锁是多个线程互相持有对方需要的资源,导致都无法继续执行。

死锁常见四个条件:

  • 互斥。
  • 请求并保持。
  • 不可剥夺。
  • 循环等待。

避免方式:

  • 固定加锁顺序。
  • 减少锁范围。
  • 使用超时锁。
  • 避免嵌套锁。
  • 使用并发工具类代替手写锁。

JVM 运行时数据区主要包括:

  • 程序计数器。
  • Java 虚拟机栈。
  • 本地方法栈。
  • 堆。
  • 方法区。

线程私有:

  • 程序计数器。
  • Java 虚拟机栈。
  • 本地方法栈。

线程共享:

  • 堆。
  • 方法区。

栈主要存放方法调用相关信息,比如局部变量表、操作数栈、方法出口等。

堆主要存放对象实例,是垃圾回收重点关注的区域。

对比项
线程关系线程私有线程共享
存储内容方法调用、局部变量对象实例
生命周期随方法调用入栈出栈由 GC 管理
常见错误StackOverflowErrorOutOfMemoryError

垃圾回收是 JVM 自动回收不再使用对象的机制。

判断对象是否可回收,主流方法是可达性分析:从 GC Roots 出发,能到达的对象是存活对象,不能到达的对象可以被回收。

常见 GC Roots:

  • 虚拟机栈中的引用。
  • 方法区中的静态变量引用。
  • 常量引用。
  • 本地方法栈中的引用。

常见算法:

  • 标记-清除:先标记垃圾,再清除,可能产生内存碎片。
  • 复制算法:把存活对象复制到另一块区域,适合新生代。
  • 标记-整理:标记后整理存活对象,减少碎片。
  • 分代收集:按对象生命周期分区,不同区域用不同算法。

OOM 是内存不足导致的错误。

常见原因:

  • 创建了大量对象,堆空间不足。
  • 大对象过多。
  • 内存泄漏,旧对象一直被引用。
  • 线程过多导致栈空间不足。
  • 元空间加载类过多。

排查时通常会看日志、堆 dump、GC 情况和对象引用链。

常见关注点:

  • 堆大小:-Xms-Xmx
  • 新生代大小:-Xmn
  • 元空间大小:-XX:MetaspaceSize
  • GC 收集器选择。
  • GC 日志。
  • 停顿时间和吞吐量。

调优不是盲目改参数,而是先看现象:内存是否够、GC 是否频繁、停顿是否过长、对象是否异常增长。

IoC 是控制反转。

原本对象由程序自己创建和管理,现在交给 Spring 容器创建和管理。

DI 依赖注入是 IoC 的一种实现方式。比如一个 Service 依赖 Mapper,不需要自己 new,而是由 Spring 注入。

AOP 是面向切面编程,用来把通用逻辑从业务代码中抽离出来。

常见场景:

  • 日志。
  • 权限校验。
  • 事务。
  • 监控统计。
  • 接口耗时。

核心思想是:不修改业务方法本身,在方法执行前后织入增强逻辑。

简化流程:

实例化 -> 属性注入 -> 初始化前后处理 -> 初始化方法 -> 使用 -> 销毁

常见扩展点:

  • 构造方法。
  • 属性填充。
  • BeanPostProcessor
  • InitializingBeaninit-method
  • DisposableBeandestroy-method

Spring Boot 主要解决传统 Spring 项目配置繁琐的问题。

优势:

  • 自动配置。
  • 内置 Web 容器。
  • starter 依赖简化。
  • 快速创建项目。
  • 方便监控和部署。

一句话回答:

Spring Boot 让 Spring 项目更容易启动和维护,它通过自动配置和 starter 机制减少大量 XML 或手动配置。

自动配置的核心是根据类路径、配置文件和条件注解,自动创建合适的 Bean。

常见关键词:

  • starter。
  • auto configuration。
  • @EnableAutoConfiguration
  • 条件注解,比如 @ConditionalOnClass@ConditionalOnMissingBean

面试中可以这样说:

Spring Boot 会根据引入的依赖和当前环境判断是否满足条件,如果满足,就把对应配置类里的 Bean 注册到容器中。

RESTful API 是一种接口设计风格。

核心思想:

  • 使用 URL 表示资源。
  • 使用 HTTP 方法表示操作。
  • 使用状态码表示结果。

示例:

GET /users/1 查询用户
POST /users 创建用户
PUT /users/1 更新用户
DELETE /users/1 删除用户

索引是帮助 MySQL 快速查找数据的数据结构。

可以理解为书的目录。没有索引时,数据库可能要全表扫描;有索引时,可以更快定位数据。

索引能提高查询效率,但会增加写入成本和存储空间。

B+Tree 适合磁盘存储和范围查询。

原因:

  • 树高度低,减少磁盘 IO。
  • 非叶子节点只存索引,能放更多 key。
  • 叶子节点之间有链表,范围查询效率高。
  • 查询性能稳定。

覆盖索引指查询需要的字段都能从索引中获得,不需要回表查询。

比如有联合索引 (name, age)

SELECT name, age FROM user WHERE name = 'xiaoxi';

如果查询字段都在索引里,就可能走覆盖索引。

常见情况:

  • 对索引列使用函数。
  • 对索引列做计算。
  • 使用左模糊匹配,例如 LIKE '%abc'
  • 联合索引不符合最左前缀原则。
  • 隐式类型转换。
  • OR 条件使用不当。

事务是一组数据库操作的集合,要么全部成功,要么全部失败。

比如下单时:

  • 创建订单。
  • 扣减库存。
  • 扣减余额。

这些操作应该作为一个整体处理。

特性含义
Atomicity 原子性事务要么全部成功,要么全部失败
Consistency 一致性事务前后数据满足约束
Isolation 隔离性并发事务之间互相隔离
Durability 持久性事务提交后数据持久保存

MVCC 是多版本并发控制。

它通过保存数据的多个版本,让读写尽量不互相阻塞。

在 InnoDB 中,MVCC 主要依赖:

  • 隐藏字段。
  • undo log。
  • ReadView。

常见作用是支持可重复读和快照读。

常见原因:

  • 数据主要在内存中。
  • 使用高效数据结构。
  • 单线程命令执行避免大量锁竞争。
  • IO 多路复用。
  • C 语言实现,执行效率高。

常见类型:

  • String。
  • Hash。
  • List。
  • Set。
  • Sorted Set。
  • Bitmap。
  • HyperLogLog。
  • Geo。
  • Stream。

缓存穿透是请求的数据在缓存和数据库中都不存在,导致请求持续打到数据库。

解决方式:

  • 缓存空对象。
  • 布隆过滤器。

缓存击穿是热点 Key 过期瞬间,大量并发请求同时打到数据库。

解决方式:

  • 互斥锁。
  • 逻辑过期。

缓存雪崩是大量 Key 同时过期,或者 Redis 整体不可用,导致大量请求涌向数据库。

解决方式:

  • 过期时间加随机值。
  • 多级缓存。
  • 限流降级。
  • Redis 高可用。

Elasticsearch 是一个分布式搜索和分析引擎,常用于全文搜索、日志检索、商品搜索等场景。

它底层基于 Lucene,对外提供 REST API,支持分布式、倒排索引和复杂查询。

倒排索引是从词到文档的映射。

普通索引更像:

文档 -> 包含哪些词

倒排索引更像:

词 -> 出现在哪些文档中

这就是 ES 做全文搜索快的重要原因。

index 类似一类数据的集合。

document 是一条 JSON 数据。

type 在早期版本中用于区分类型,但新版本已经逐步移除,不建议在新项目中依赖 type。

可以类比:

ES关系型数据库
indextable 或 database 的某种集合概念
documentrow
fieldcolumn

DSL 是 Elasticsearch 的查询语言,使用 JSON 描述查询条件。

示例:

es-dsl.json
{
"query": {
"match": {
"title": "redis"
}
}
}

常见条件:

  • must:必须匹配,影响评分。
  • should:可选匹配,可能影响评分。
  • filter:必须匹配,但不参与评分,适合过滤条件。
  • must_not:必须不匹配。

为什么 Elasticsearch 搜索比 MySQL 快

Section titled “为什么 Elasticsearch 搜索比 MySQL 快”

ES 在全文搜索场景下更快,主要因为它使用倒排索引。

MySQL 更擅长结构化数据查询和事务处理,全文检索不是它最核心的场景。

可以这样回答:

ES 会先对文本分词,再建立词到文档的倒排索引。查询关键词时,可以快速定位包含该词的文档。MySQL B+Tree 索引更适合精确匹配和范围查询,对复杂全文搜索、相关性评分和分词检索不如 ES 合适。

Docker 是容器化技术,可以把应用和依赖打包成镜像,再以容器方式运行。

它解决了“本地能跑,服务器不能跑”的环境一致性问题。

对比项Docker虚拟机
隔离方式进程级隔离硬件级虚拟化
启动速度
资源占用较少较多
系统内核共享宿主机内核每个虚拟机有完整操作系统
适用场景应用部署、微服务强隔离、多系统环境

常见命令:

linux-commands.sh
ls
cd
pwd
mkdir
rm
cp
mv
cat
tail -f app.log
grep "error" app.log
ps -ef
top
df -h
free -m
chmod
chown

面试时如果结合项目部署经历回答,会比单纯背命令更好。

REST API 是基于 REST 风格设计的接口。

它通常使用 HTTP 协议,通过 URL 表示资源,通过 HTTP 方法表示操作。

这个问题和 Spring 里的 RESTful API 本质相同。

微服务是把一个大系统拆成多个小服务,每个服务负责一个相对独立的业务能力。

优点:

  • 服务独立部署。
  • 技术栈可以更灵活。
  • 方便水平扩展。
  • 团队边界更清晰。

缺点:

  • 服务间调用复杂。
  • 分布式事务困难。
  • 监控、链路追踪、部署复杂度提高。

API 网关是系统入口,负责把外部请求转发到内部服务。

常见功能:

  • 路由转发。
  • 鉴权。
  • 限流。
  • 熔断。
  • 日志。
  • 跨域处理。

如果你的项目重点是 Elasticsearch,面试官很可能会围绕项目继续追问。

可以提前准备这些问题:

问题准备方向
为什么项目要用 ESMySQL 搜索能力不足、全文检索、分词、排序、性能
数据怎么同步到 ES同步写、异步消息、定时补偿
ES 和 MySQL 数据不一致怎么办重试、补偿任务、最终一致性
索引怎么设计index、mapping、分词器、字段类型
搜索结果怎么排序相关性评分、业务权重、时间、热度
查询慢怎么优化filter、分页限制、字段选择、索引设计

面试时不要只说“我用了 ES”。更好的说法是:

我在项目中用 Elasticsearch 解决全文检索问题。MySQL 更适合事务和结构化查询,但对分词搜索、相关性排序支持有限。所以把需要搜索的数据同步到 ES,通过倒排索引提升搜索效率。项目里还需要考虑 MySQL 和 ES 的数据一致性,比如通过消息队列异步同步,并配合定时任务补偿。

学习能力问题通常不会太难,但很考验真实感。

常见问题:

  • 最近在学什么技术?
  • 为什么选择 Java?
  • 遇到不会的问题怎么解决?
  • 看过哪些技术文档?
  • 项目里最有收获的地方是什么?

回答建议:

  • 不要只说“我在学 Java”。
  • 要说具体学了什么、为什么学、怎么实践。
  • 最好能和项目或面试岗位关联起来。

例如:

最近我在补 Redis 和 Elasticsearch。Redis 主要关注缓存穿透、击穿、雪崩以及数据结构的使用场景;Elasticsearch 主要学习倒排索引、DSL 查询和数据同步。因为我的项目里有搜索场景,所以我想把搜索链路和缓存链路都理解得更完整。

如果时间有限,可以按这个顺序准备:

  1. Java 基础:集合、字符串、异常、反射。
  2. MySQL:索引、事务、MVCC、索引失效。
  3. Redis:数据结构、缓存问题、常用命令。
  4. Spring Boot:IoC、AOP、自动配置、REST API。
  5. JVM:内存结构、GC、OOM。
  6. 并发:线程池、锁、volatile、死锁。
  7. Elasticsearch 项目:倒排索引、DSL、数据同步、搜索优化。
  8. Docker 和 Linux:能讲清楚部署和常用命令即可。

Java 校招面试不是只背八股。

基础题要能答清楚概念,项目题要能讲清楚自己做了什么、为什么这样做、遇到了什么问题、怎么解决。

这份清单的重点不是一次性背完,而是把每个问题都整理成三层:

一句话结论
核心原理
项目或使用场景

这样回答时会更稳,也更像真的理解过。

爬虫与 JS 逆向面试题复盘

这是一组爬虫和 JS 逆向相关的面试题。

这类面试不会只问“会不会用 requests”,更常见的是从一个具体业务场景开始追问:怎么登录、怎么抓动态接口、怎么处理反爬、百万级数据怎么调度、数据怎么进入后续处理链路。

这篇文章按面试题复盘的方式整理,重点是把回答讲得更清楚、更工程化。

同时要先明确一点:爬虫和逆向要遵守法律、站点协议和数据合规要求。面试中可以讲技术思路,但不应该表达绕过风控、攻击站点或采集敏感数据的意图。更稳的说法是:在授权范围内做数据采集和接口分析。

题目:

登录有两种方式,一种是账号密码登录,并且需要输入动态 token;另一种是二维码登录。如果要自动登录,你会选择哪种方式,为什么?说说实现方法。

我的回答倾向是:优先选择账号密码加 token 的方式。

原因:

  • 账号密码登录更适合程序化请求。
  • 登录流程相对稳定,便于抓包分析。
  • 登录成功后可以拿到 token、cookie 或 session。
  • 二维码登录通常依赖人工扫码,不适合长期自动化任务。

可以这样回答:

我会优先选择账号密码加 token 的登录方式。因为它更容易通过请求和响应模拟,流程上可以先请求登录页或初始化接口,拿到登录所需的 token、cookie,再携带账号密码和动态 token 请求登录接口。登录成功后保存 cookie 或 access token,后续请求统一带上认证信息。二维码登录更适合人工确认,自动化成本更高,而且很多二维码登录会绑定设备、时效和扫码确认,不适合爬虫任务长期稳定运行。

一个简化流程:

请求登录页或初始化接口
|
获取 token / csrf / cookie
|
提交账号、密码、动态 token
|
登录成功
|
保存 cookie 或 access token
|
后续请求携带认证信息

需要注意:如果动态 token 是验证码、短信码、二次验证,不能假设可以无成本自动化。面试里可以强调“在授权账号和合规场景下处理登录态”。

题目:

这种前端返回数据的网站,如何爬取数据?

现在很多网站是前端框架渲染,HTML 源码里没有完整数据。此时不要急着解析页面,而是先看网络请求。

常规步骤:

  1. 打开 Chrome DevTools。
  2. 进入 Network 面板。
  3. 过滤 XHR / Fetch 请求。
  4. 找到真正返回 JSON 数据的接口。
  5. 分析 URL、请求方法、参数、Headers、Cookie。
  6. 用 Python 模拟请求。
fetch-api-data.py
import requests
url = "https://example.com/api/list"
headers = {
"User-Agent": "Mozilla/5.0",
"Referer": "https://example.com/list",
}
params = {
"page": 1,
"size": 20,
}
response = requests.get(url, headers=headers, params=params, timeout=10)
data = response.json()
print(data)

如果接口参数是动态生成的,就继续分析 JS。

如果页面确实没有接口,或者数据必须通过浏览器运行后才出现,可以考虑 Selenium 或 Playwright。但大规模采集时,优先分析接口,因为浏览器自动化成本更高。

题目:

面对百万甚至千万数据量的爬取,你的爬取策略是怎么样的?爬取到的数据如何存储?

这题考察的是系统设计,而不是单机脚本。

可以从四层回答:

  • 任务拆分。
  • 并发控制。
  • 反爬与容错。
  • 数据存储和后续处理。

一个比较完整的链路是:

任务调度 -> 爬虫采集 -> Kafka -> Flink 清洗 -> 数据存储

如果只是普通项目,可以存 MySQL 或 CSV;如果是百万、千万规模,就要考虑分批写入、去重、失败重试、数据清洗和存储扩展。

可以这样回答:

面对百万级数据,我不会用一个单机脚本顺序爬。一般会先把任务拆成分页任务、关键词任务或 ID 区间任务,放到任务队列里,由多个爬虫节点并发消费。采集时会限制请求频率,设置超时重试和代理池,避免单点 IP 或账号压力过大。采集到的数据先进入 Kafka,后续由 Flink 做实时清洗,再写入 MySQL、ES 或数据仓库。对于失败任务会记录状态,后续补偿重试。

常用 Chrome DevTools 的 Network 面板。

主要看:

  • XHR / Fetch 请求。
  • 请求 URL。
  • 请求方法。
  • Query 参数和 Request Payload。
  • Headers。
  • Cookie。
  • Response。
  • Initiator 调用来源。

如果参数是动态生成的,会继续去 Sources 面板断点调试,或在 JS 文件中搜索参数名。

可以从几个现象判断:

  • 请求参数里存在动态加密参数。
  • 接口依赖 token、cookie、签名或时间戳。
  • 请求频率过高会被封 IP。
  • 返回内容出现验证码、空数据或风控页面。
  • 同一个接口在浏览器能访问,程序请求失败。
  • Headers 缺失时返回异常。

可以这样回答:

我会先比较浏览器正常请求和程序模拟请求的差异。如果同样的 URL 在浏览器里返回正常,但程序里返回空数据、验证码、403 或风控响应,就说明可能存在反爬。再继续分析是否有动态参数、token 校验、cookie 校验、频率限制或行为检测。

XPath 和 CSS Selector 都能定位 HTML 节点。

对比项XPathCSS Selector
语法类似路径表达式类似 CSS 选择器
能力更强,支持轴、文本、复杂路径简洁,适合常见选择
可读性复杂表达式可读性一般简单场景更清晰
爬虫常用度很常用也常用

面试可以说:

简单页面我会用 CSS Selector,因为语法简洁;复杂定位,比如按文本、层级、相邻节点查找时,我更倾向 XPath。

可以按流程回答:

  1. 使用 Network 抓包找到目标接口。
  2. 确认哪个参数是动态生成的。
  3. 全局搜索参数名。
  4. 在 Sources 面板下断点。
  5. 观察 Call Stack 调用链。
  6. 找到最终生成参数的函数。
  7. 用 Python 或 Node 复现算法。

更完整的回答:

我在项目中遇到过接口参数由 JS 加密生成的情况。处理时先通过 Network 找到接口和异常参数,然后在 JS 文件中搜索参数名。如果搜索不到,就从请求发起位置或 XHR 断点入手,在 Sources 里下断点,结合 Call Stack 分析调用链,找到参数生成函数。确定算法后,再用 Python 或 Node 复现,最后和浏览器生成结果对比,确保请求参数一致。

这类回答要强调“分析和复现授权接口参数”,不要说成攻击或绕过安全系统。

JS 混淆后,变量名和函数名可能没有意义,所以不要期待完全看懂所有代码。

常见思路:

  • 不全量还原,只找关键链路。
  • 通过 XHR/fetch 断点定位请求发起位置。
  • 使用 Call Stack 看调用链。
  • 打印关键变量。
  • 对关键函数做输入输出对比。
  • 必要时把关键函数拎出来运行。

可以这样回答:

遇到混淆 JS 时,我不会从头读完整文件,而是围绕目标接口定位关键参数。通过断点、调用栈、关键变量打印和函数输入输出分析,逐步缩小范围,最终定位生成参数的函数。

Selenium 是浏览器自动化工具,适合复杂页面、需要真实浏览器环境的场景。

但它的问题也明显:

  • 启动浏览器成本高。
  • 并发能力弱。
  • 资源占用大。
  • 速度慢。
  • 大规模采集不划算。

所以一般优先分析接口直接请求。只有接口很难复现、页面强依赖浏览器环境、或需要真实交互时,才考虑 Selenium 或 Playwright。

面试回答:

Selenium 可以用,但我不会作为首选。因为大规模采集更关注吞吐和稳定性,直接请求接口效率更高。Selenium 更适合登录、复杂交互或无法绕开浏览器渲染的页面。

长期稳定运行靠的不是一个脚本,而是容错和监控。

常见机制:

  • 请求超时。
  • 失败重试。
  • 指数退避。
  • 异常捕获。
  • 失败任务记录。
  • 账号状态检测。
  • IP 或代理状态检测。
  • 任务监控。
  • 健康检查。
  • 失败告警。

可以这样回答:

我会为爬虫设计超时重试、异常捕获、失败任务记录和任务监控机制。如果请求失败,会根据错误类型决定重试、切换账号、切换代理或标记任务失败。系统层面会有健康监测和失败上报,保证爬虫可以长期稳定运行。

如果项目里支持 500万+ / 日 的采集规模,可以这样回答:

系统支持 500 万以上日采集量。采集任务不是由单个脚本完成,而是通过任务调度系统统一拆分和分发,多节点并发执行。采集结果进入 Kafka,再由 Flink 进行实时清洗和处理。

面试时不要只报数字,最好补上支撑数字的架构。

整体链路可以这样描述:

调度系统
|
v
爬虫节点
|
v
Kafka
|
v
Flink
|
v
MySQL / Elasticsearch / 数据仓库

各模块职责:

模块作用
调度系统生成任务、分配任务、协调账号
爬虫节点执行采集、解析数据、处理重试
Kafka解耦采集和处理,缓冲流量
Flink实时清洗、过滤、转换
存储层存储清洗后的业务数据

这类回答会比“我用 Scrapy 分布式”更有工程感。

调度系统主要负责任务生成和账号协调。

你笔记中的规模是:

  • 1400+ 爬虫任务。
  • 400+ 账号 Cookie。
  • 任务信息存储在 Redis。

可以这样回答:

调度系统会把采集目标拆成具体任务,任务状态存储在 Redis 中。爬虫节点从 Redis 获取任务,执行后回写任务状态。账号 Cookie 也由调度系统统一管理,分配任务时会根据账号状态选择可用账号,避免单个账号压力过大。

Redis 适合做任务队列和状态缓存。

原因:

  • 读写性能高。
  • 支持 List、Set、Hash、Sorted Set 等结构。
  • 适合存任务状态、账号状态和临时调度数据。
  • 操作简单,延迟低。

可以补一句:

如果任务需要更强的可靠性、确认机制和重试语义,也可以引入消息队列;Redis 更适合轻量级任务调度和状态管理。

账号失效的表现:

  • 登录失败。
  • Cookie 失效。
  • 返回 401、403。
  • 返回验证码或风控页面。
  • 请求结果为空或异常。

处理方式:

  • 标记账号不可用。
  • 暂停该账号任务。
  • 重新调度任务。
  • 切换可用账号。
  • 触发重新登录或人工处理。

可以这样回答:

系统会根据响应状态和页面内容判断账号是否异常。一旦发现 Cookie 失效或登录状态异常,就标记账号状态,避免继续分配任务,同时把未完成任务重新放回队列,交给其他可用账号处理。

常用库:

  • requests:发送 HTTP 请求。
  • httpx:支持同步和异步请求。
  • scrapy:爬虫框架。
  • lxml:解析 HTML,支持 XPath。
  • beautifulsoup4:HTML 解析。
  • selenium:浏览器自动化。
  • playwright:现代浏览器自动化。

项目里如果主要使用 requests + XPath,可以这样说:

普通接口采集我主要使用 requests,请求接口后用 XPath 或 JSON 解析数据。如果是复杂任务调度和大规模采集,会考虑 Scrapy 或自研调度系统。

基本做法:

request-timeout.py
import requests
try:
response = requests.get(
"https://example.com/api",
timeout=(3, 10),
)
response.raise_for_status()
except requests.Timeout:
# 记录超时并重试
pass
except requests.RequestException:
# 记录其他请求异常
pass

可以配合:

  • 固定次数重试。
  • 指数退避。
  • 失败任务入库。
  • 切换代理或账号。

常见方式:

  • 设置请求间隔。
  • 限制并发数量。
  • 使用任务队列控制消费速度。
  • 对单域名限速。
  • 对单账号限速。
  • 对异常响应动态降速。

面试里可以说:

控制速度不只是 sleep,而是结合并发数、任务队列、账号维度和站点响应来动态调整,避免触发反爬,也保护目标站点和自身系统。

Docker 的价值:

  • 保证运行环境一致。
  • 方便部署。
  • 方便横向扩展多个爬虫节点。
  • 便于隔离依赖。
  • 适合配合 CI/CD。

爬虫系统里尤其适合把爬虫节点容器化。需要扩容时,可以快速启动多个容器实例。

Kafka 主要承担数据通道和缓冲层。

作用:

  • 解耦采集和处理。
  • 缓冲高峰流量。
  • 支持高吞吐数据传输。
  • 方便后续多个消费者处理数据。

可以这样回答:

爬虫采集速度和后续清洗入库速度不一定一致,所以中间用 Kafka 解耦。爬虫只负责把原始数据写入 Kafka,Flink 再从 Kafka 消费并清洗处理。

Redis 在项目中可以承担:

  • 任务队列。
  • 任务状态缓存。
  • 账号 Cookie 管理。
  • 去重集合。
  • 临时失败记录。
  • 限速计数。

面试回答:

Redis 主要用于调度层,保存任务队列、任务状态和账号 Cookie。因为它读写快,并且数据结构丰富,适合管理这种高频变化的临时状态。

可以回答 JS 加密参数逆向。

更完整的说法:

最难的是 JS 加密参数逆向。因为网站 JS 做了混淆,不能直接通过阅读代码看懂逻辑。我通过 Network 定位接口和动态参数,再用 Sources 下断点,结合调用栈分析参数生成流程,最后把关键算法用 Python 或 Node 复现出来。这个过程比较考验调试能力和耐心。

处理步骤:

  1. 先复现问题,确认是哪些请求失败。
  2. 对比正常浏览器请求和爬虫请求差异。
  3. 判断新增机制:token、cookie、签名、频率、验证码、行为检测。
  4. 如果是参数变化,重新调试 JS。
  5. 如果是频率问题,调整限速和调度策略。
  6. 如果涉及强验证或合规风险,停止采集或走授权接口。

可以这样回答:

我会先分析新增反爬属于哪一类,再决定策略。如果是参数签名变化,就重新定位 JS 生成逻辑;如果是频率限制,就降低并发、调整账号和代理策略;如果是登录态或 Cookie 变化,就更新账号状态检测和重新登录流程。对于验证码或强风控场景,需要评估合规性,不能盲目绕过。

如果面试官让你整体介绍这个爬虫项目,可以这样组织:

这个项目主要做大规模数据采集。整体链路是爬虫采集、Kafka 缓冲、Flink 清洗、最终写入存储。爬虫侧通过 Chrome DevTools 分析接口,优先直接请求接口而不是 Selenium。调度系统负责管理 1400 多个任务和 400 多个账号 Cookie,任务状态存储在 Redis。系统支持超时重试、失败任务记录、账号失效检测和健康监控。项目中比较难的是 JS 加密参数逆向,我通过断点调试、调用栈分析和算法复现解决过接口动态参数问题。

爬虫和逆向面试题,重点不是只会某个库,而是能把采集链路讲完整:

接口分析 -> 登录态处理 -> 参数逆向 -> 任务调度 -> 并发控制 -> 数据通道 -> 清洗入库 -> 监控补偿

如果能把这条链路讲清楚,再结合自己实际做过的规模、账号调度、Kafka/Flink、Redis 和 Docker,回答就会更像真实项目经验,而不是零散知识点。

一道 gRPC 简易计算器面试题复盘

这是我曾经做过的一道面试题中的一个小项目:用 Go + gRPC + gRPC-Web + Next.js 做一个简易计算器。

题目本身不复杂,只有加、减、乘、除四种运算。但它真正考察的不是“会不会写计算器”,而是能不能把前后端通信、接口契约、错误处理和跨端调用讲清楚。

项目地址:GolangNextGrpcSimpleCalculator

这道题可以理解为:

前端输入两个数字和一个运算符,通过 gRPC-Web 调用 Go 后端,后端完成计算并返回结果。

最小功能闭环包括:

  • 前端页面可以输入两个操作数。
  • 前端可以选择 +-*/ 四种运算符。
  • 前端通过 gRPC-Web 调用后端。
  • 后端使用 Go 实现 gRPC 服务。
  • 后端根据请求参数返回计算结果。
  • 除数为 0 或未知运算符时,需要返回错误。

如果只是做一个普通 HTTP 接口,这题可能很快就结束了。但这里要求使用 gRPC,就会多出一层“接口契约”的设计:前后端都要围绕 .proto 文件生成代码。

这个项目大致可以拆成两部分:

  • 文件夹calculator-backend/
    • 文件夹calculator/
      • calculator.proto
    • main.go
    • calculator_test.go
    • go.mod
  • 文件夹calculator-frontend/
    • 文件夹app/
      • 文件夹generated/
        • 文件夹calculator/
      • page.tsx
    • 文件夹calculator/
      • calculator.proto
    • package.json

后端负责定义和实现计算服务,前端负责生成 gRPC-Web 客户端代码并发起调用。

这里最核心的文件有三个:

文件作用
calculator.proto定义服务、请求结构、响应结构
main.go实现 Go gRPC 服务,并包装成 gRPC-Web HTTP 服务
page.tsx在 Next.js 页面中创建请求并调用后端

gRPC 的入口通常不是先写控制器,而是先写 .proto

calculator.proto
syntax = "proto3";
package calculator;
option go_package = "github.com/2760439882/calculator-backend/calculator;calculator";
service Calculator {
rpc Calculate(CalculationRequest) returns (CalculationResponse);
}
message CalculationRequest {
double operand1 = 1;
double operand2 = 2;
string operator = 3; // "+", "-", "*", "/"
}
message CalculationResponse {
double result = 1;
}

这份协议里有三个关键信息:

  • Calculator 是服务名。
  • Calculate 是远程调用方法。
  • CalculationRequestCalculationResponse 分别是请求和响应结构。

这就是 gRPC 和普通 REST 接口很不一样的地方。REST 接口经常先约定 URL 和 JSON 字段,而 gRPC 会先约定服务方法和强类型消息结构。

后端使用 Go 实现 Calculator 服务。

main.go
type server struct {
pb.UnimplementedCalculatorServer
}
func (s *server) Calculate(
ctx context.Context,
req *pb.CalculationRequest,
) (*pb.CalculationResponse, error) {
var result float64
switch req.Operator {
case "+":
result = req.Operand1 + req.Operand2
case "-":
result = req.Operand1 - req.Operand2
case "*":
result = req.Operand1 * req.Operand2
case "/":
if req.Operand2 == 0 {
return nil, fmt.Errorf("division by zero")
}
result = req.Operand1 / req.Operand2
default:
return nil, fmt.Errorf("unknown operator")
}
return &pb.CalculationResponse{Result: result}, nil
}

这段实现很适合面试时讲,因为它虽然简单,但覆盖了服务端接口实现的几个基本点:

  • 方法签名来自 .proto 生成代码。
  • 请求参数来自 CalculationRequest
  • 返回值必须符合 CalculationResponse
  • 错误可以通过 error 返回给调用方。
  • 除法需要额外处理除数为 0 的情况。

真正的业务逻辑只有一个 switch,但重点是:这个 switch 被放在了 gRPC 服务方法里,前端不会直接知道后端怎么计算,只知道自己要调用 Calculate

浏览器不能像 Go、Java、Node 服务端那样直接发起原生 gRPC 请求。原生 gRPC 基于 HTTP/2,而浏览器端直接使用 gRPC 会受到限制。

所以前端调用 gRPC 服务时,通常需要一层 gRPC-Web。

这个项目里,Go 后端把原始 gRPC 服务包装成了 gRPC-Web 服务:

main.go
grpcServer := grpc.NewServer()
pb.RegisterCalculatorServer(grpcServer, &server{})
wrappedGrpc := grpcweb.WrapServer(grpcServer)
httpServer := http.Server{
Addr: ":8080",
Handler: cors.New(cors.Options{
AllowedOrigins: []string{"http://localhost:3000"},
AllowedMethods: []string{"GET", "POST", "OPTIONS"},
AllowedHeaders: []string{
"Content-Type",
"X-Grpc-Web",
"X-User-Agent",
"grpc-timeout",
},
AllowCredentials: true,
}).Handler(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
if wrappedGrpc.IsGrpcWebRequest(r) ||
wrappedGrpc.IsAcceptableGrpcCorsRequest(r) ||
wrappedGrpc.IsGrpcWebSocketRequest(r) {
wrappedGrpc.ServeHTTP(w, r)
return
}
http.NotFound(w, r)
})),
}

这里有两个面试时值得说清楚的点。

第一,后端并不是直接写一个普通 HTTP JSON 接口,而是先创建 grpc.NewServer(),再通过 grpcweb.WrapServer() 包装。

第二,因为前端运行在 http://localhost:3000,后端运行在 http://localhost:8080,所以需要配置 CORS。否则浏览器会先把请求挡掉,根本到不了 gRPC-Web 服务。

前端是 Next.js 页面。因为要在浏览器里响应用户输入和发起请求,所以页面文件使用了 'use client'

page.tsx
'use client';
import { useState } from 'react';
import { CalculatorClient } from './generated/calculator/calculator_grpc_web_pb';
import { CalculateRequest } from './generated/calculator/calculator_pb';
const client = new CalculatorClient('http://localhost:8080', null, null);
export default function Home() {
const [operand1, setOperand1] = useState('');
const [operand2, setOperand2] = useState('');
const [operator, setOperator] = useState('+');
const [result, setResult] = useState<number | null>(null);
const [error, setError] = useState<string | null>(null);
const [loading, setLoading] = useState(false);
const handleCalculate = () => {
setLoading(true);
setResult(null);
setError(null);
const req = new CalculateRequest();
req.setOperand1(parseFloat(operand1));
req.setOperand2(parseFloat(operand2));
req.setOperator(operator);
client.calculate(req, {}, (err, response) => {
setLoading(false);
if (err) {
setError('请求失败: ' + err.message);
return;
}
setResult(response.getResult());
});
};
}

这里的关键不是 useState,而是这三步:

  1. 使用生成的 CalculatorClient 创建客户端。
  2. 使用生成的 CalculateRequest 创建请求对象。
  3. 调用 client.calculate(),在回调里处理结果或错误。

这说明前端并没有手写请求路径、请求体字段和响应解析逻辑。它依赖 .proto 生成的代码来保证调用结构一致。

如果把这道题当成面试题看,它主要考察下面几类能力。

考点具体体现
gRPC 基础是否知道 .proto、service、message、生成代码
前后端通信是否知道浏览器不能直接使用原生 gRPC,需要 gRPC-Web
Go 服务端是否能注册服务、实现接口、启动服务
错误处理是否处理除数为 0、未知运算符、请求失败
跨域问题是否知道前后端端口不同会触发 CORS
工程意识是否能把生成代码、后端代码、前端代码分清楚

所以面试时不要只说“我实现了一个计算器”。更好的说法是:

我用 .proto 定义了计算服务和消息结构,再用 Go 实现 gRPC 服务。因为前端运行在浏览器里,不能直接调用原生 gRPC,所以后端用 gRPC-Web 包装了一层 HTTP 服务,并配置 CORS。前端通过生成的 gRPC-Web Client 创建请求对象,调用后端的 Calculate 方法,最后处理成功结果和错误信息。

这段回答会比“我写了加减乘除”更能说明你理解了题目。

这个项目作为面试题已经能跑通主流程,但如果继续完善,可以从下面几个方向补强。

第一,输入校验可以提前放在前端。

现在前端直接对输入值 parseFloat()。如果用户没有输入数字,可能得到 NaN。可以在发请求前判断两个操作数是否合法。

第二,后端错误可以使用 gRPC status。

当前代码使用 fmt.Errorf() 返回错误。实际项目里可以使用 status.Error() 搭配 codes.InvalidArgument,这样调用方可以更准确地区分错误类型。

return nil, status.Error(codes.InvalidArgument, "division by zero")

第三,运算符可以使用枚举。

当前 operator 是字符串,优点是直观,缺点是容易传入非法值。如果要更严谨,可以在 .proto 中定义 enum。

enum Operator {
OPERATOR_UNSPECIFIED = 0;
ADD = 1;
SUBTRACT = 2;
MULTIPLY = 3;
DIVIDE = 4;
}

第四,配置可以抽出来。

前端里的 http://localhost:8080 和后端 CORS 里的 http://localhost:3000 都是开发环境地址。后续如果部署,最好放进环境变量里。

这道题可以按四步讲:

  1. 先说题目目标:前端输入表达式,后端通过 gRPC 完成计算。
  2. 再说接口契约:用 .proto 定义 Calculate 方法、请求和响应。
  3. 接着说后端实现:Go 实现服务,处理四种运算和异常情况。
  4. 最后说浏览器调用:使用 gRPC-Web 生成客户端,前端发起请求并处理结果。

如果面试官继续追问,可以展开这几个点:

  • gRPC 和 REST 的区别是什么?
  • 为什么浏览器需要 gRPC-Web?
  • .proto 文件改了以后要做什么?
  • 如果除数为 0,应该怎么返回错误?
  • 如果以后要支持更多运算,应该怎么扩展?

这个项目的价值不在于计算器本身,而在于它用一个很小的功能,把 gRPC 项目里最关键的链路串起来了:

proto 定义 -> 生成代码 -> Go 实现服务 -> gRPC-Web 包装 -> 前端调用 -> 展示结果

对于面试题来说,这种项目很合适。它足够小,能在有限时间内完成;同时又能覆盖接口设计、后端实现、前端调用、跨域和错误处理这些真实工程里会遇到的问题。

Java 集合底层:List、Set、HashMap 与红黑树

Java 集合面试题经常会从一个很小的问题开始:

List 和 Set 有什么区别?

如果继续追问,很快就会进入 HashMapHashSetTreeSet、红黑树、哈希冲突、扩容、equals()hashCode() 这些内容。

这篇文章把几份旧笔记整理成一条完整的复习线:先看集合接口,再看哈希表,最后看树结构。

ListSet 都继承自 Collection 接口,都属于 Java 集合体系中存放单个元素的容器。

它们的核心区别可以从三个角度理解。

List 允许重复元素:

List<String> list = new ArrayList<>();
list.add("Java");
list.add("Java");
System.out.println(list.size()); // 2

Set 不允许重复元素:

Set<String> set = new HashSet<>();
set.add("Java");
set.add("Java");
System.out.println(set.size()); // 1

需要注意的是,Set 判断重复通常依赖 equals()hashCode()。如果自定义对象放入 HashSet,却没有正确重写这两个方法,就可能出现“看起来相同的对象却没有去重”的问题。

List 按插入顺序保存元素,可以通过下标访问:

list.get(0);

Set 不一定保持插入顺序,也不能通过下标访问元素。

不同 Set 实现的顺序语义也不同:

  • HashSet:不保证插入顺序,也不保证排序。
  • LinkedHashSet:按插入顺序迭代。
  • TreeSet:按自然顺序或比较器规则排序。

所以不能简单说“Set 会升序排序”。准确说法是:TreeSet 会排序,HashSet 不保证顺序。

List 更像动态数组或链表,具体性能取决于实现类。

ArrayList 的特点:

  • 按下标访问快,时间复杂度通常是 O(1)
  • 中间插入和删除可能需要移动元素,时间复杂度通常是 O(n)

LinkedList 的特点:

  • 插入和删除节点时,只需要调整节点引用。
  • 但查找指定位置的元素需要遍历链表。

Set 更强调“唯一性”。例如 HashSet 底层依赖哈希表,添加、删除、查找在理想情况下可以接近 O(1)

CollectionListSet 等集合接口的上层接口,常见方法包括:

boolean add(E e);
boolean remove(Object o);
boolean contains(Object o);
boolean isEmpty();
int size();
void clear();
Iterator<E> iterator();
Object[] toArray();

有一个细节很容易忽略:Collection 本身没有 get(index) 方法。

因为不是所有集合都有下标语义。例如 List 可以按下标取值,但 Set 没有稳定的下标概念,只能通过迭代器遍历:

for (String item : set) {
System.out.println(item);
}

哈希表也叫散列表,是一种非常重要的数据结构。很多缓存、字典、索引、去重结构的核心思想,都是在内存中维护一张哈希表。

哈希表的核心公式可以理解为:

存储位置 = f(关键字)

这里的 f 就是哈希函数。它会根据 key 计算出一个哈希值,再根据数组长度换算成数组下标。

理想情况下,哈希表的查询过程是:

key -> hash -> 数组下标 -> 找到元素

如果没有哈希冲突,查找、插入、删除都可以接近 O(1)

和其他结构对比:

  • 数组:按下标访问快,按值查找通常需要遍历。
  • 链表:插入删除节点方便,但查找需要遍历。
  • 平衡二叉搜索树:查找、插入、删除通常是 O(log n)
  • 哈希表:理想情况下查找、插入、删除接近 O(1)

不过哈希表的性能高度依赖哈希函数、数组容量、负载因子和冲突处理方式。

哈希冲突指的是:不同 key 经过哈希计算后,落到了同一个数组位置。

例如:

keyA -> index 3
keyB -> index 3

这时就需要处理冲突。

常见方式有三类:

  1. 开放地址法:当前位置冲突后,继续寻找下一个可用位置。
  2. 再哈希法:使用另一个哈希函数重新计算位置。
  3. 链地址法:数组每个位置挂一个链表或树,把冲突元素放到同一个桶里。

Java HashMap 使用的就是链地址法的思路。更准确地说,在 Java 8 之后,HashMap 的桶结构可能是:

数组 + 链表
数组 + 红黑树

当同一个桶里的元素过多时,链表会在满足条件后树化,变成红黑树,以降低极端冲突下的查询成本。

HashMap 存储的是 key-value 键值对。

可以把它简化理解成:

HashMap
table 数组
bucket 0 -> null
bucket 1 -> Node -> Node
bucket 2 -> Node
bucket 3 -> TreeNode 红黑树

每个节点大致保存:

hash
key
value
next

put 一个元素时,大致过程是:

  1. 计算 key 的 hash。
  2. 根据 hash 和数组长度计算桶下标。
  3. 如果桶为空,直接放入。
  4. 如果桶不为空,判断 key 是否已经存在。
  5. 如果 key 已存在,更新 value。
  6. 如果 key 不存在,挂到链表或红黑树中。
  7. 如果元素数量超过阈值,触发扩容。

六、HashMap 的初始容量和负载因子

Section titled “六、HashMap 的初始容量和负载因子”

HashMap 有两个重要参数:

initialCapacity 初始容量
loadFactor 负载因子

常见默认值是:

initialCapacity = 16
loadFactor = 0.75

阈值计算方式可以理解为:

threshold = capacity * loadFactor

默认情况下:

16 * 0.75 = 12

当元素数量超过阈值后,HashMap 会扩容。扩容通常会带来重新分布元素的成本,所以如果能预估数据量,创建 HashMap 时可以指定合适的初始容量。

HashMap 的扩容通常发生在元素数量超过阈值之后:

size > threshold

而阈值由容量和负载因子决定:

threshold = capacity * loadFactor

默认情况下:

capacity = 16
loadFactor = 0.75
threshold = 12

也就是说,当第 13 个元素放入时,HashMap 就可能触发扩容。

扩容的核心动作可以概括成三步:

  1. 创建一个更大的新数组。
  2. 把旧数组中的节点迁移到新数组。
  3. 重新计算扩容后的阈值。

通常情况下,新容量会变成旧容量的 2 倍:

oldCapacity = 16
newCapacity = 32

新阈值也会随之变化:

oldThreshold = 16 * 0.75 = 12
newThreshold = 32 * 0.75 = 24

在 Java 8 之后,HashMap 扩容迁移时有一个很重要的规律:

元素的新位置要么是原位置,要么是原位置 + oldCapacity

例如旧数组长度是 16,某个元素原来在下标 5

旧位置:5
新位置:5 或 5 + 16 = 21

为什么会这样?因为数组长度翻倍后,参与下标计算的二进制位多了一位。只需要看 hash 在这一位上是 0 还是 1

hash & oldCapacity == 0 -> 留在原位置
hash & oldCapacity != 0 -> 移到原位置 + oldCapacity

可以用一个简化例子理解:

旧容量 16:下标范围 0 ~ 15
新容量 32:下标范围 0 ~ 31
原 bucket 5 拆成:
bucket 5
bucket 21

这也是 HashMap 容量使用 2 的次幂的一个好处:扩容后不需要完全重新计算每个元素的位置,可以通过位运算快速拆分旧桶。

Java 8 之后,HashMap 桶内链表过长时可能会树化为红黑树,但并不是链表一长就立刻树化。

常见条件可以记成:

桶内链表长度 >= 8
数组容量 >= 64

如果桶内链表已经很长,但数组容量还比较小,HashMap 通常会优先扩容,而不是马上树化。

原因也很直观:容量太小时,冲突可能只是数组太小导致的。先扩容可以让元素重新分布,很多冲突自然会减少。

面试里可以这样总结:

HashMap 默认容量是 16,负载因子是 0.75,超过阈值后通常扩容为原来的 2 倍。Java 8 扩容迁移时,元素的新位置要么保持原下标,要么移动到原下标加旧容量。桶内链表过长时可能树化,但容量较小时会优先扩容。

八、为什么 HashMap 容量常用 2 的次幂

Section titled “八、为什么 HashMap 容量常用 2 的次幂”

HashMap 的数组长度通常会调整为 2 的次幂。

这样做的一个重要原因是可以用位运算快速计算下标:

index = (n - 1) & hash;

n 是 2 的次幂时,n - 1 的二进制低位全是 1,可以更好地利用 hash 的低位信息。

扩容时也有一个好处:元素的新位置通常只有两种可能:

原位置
原位置 + oldCapacity

这可以减少扩容后重新计算和移动元素的成本。

九、equals 和 hashCode 为什么要一起重写

Section titled “九、equals 和 hashCode 为什么要一起重写”

HashMapHashSet 这类哈希结构里,hashCode() 决定元素大致落在哪个桶,equals() 决定桶内元素是否真的相等。

如果两个对象通过 equals() 判断相等,它们的 hashCode() 必须相等。

错误示例:

class User {
private Long id;
@Override
public boolean equals(Object obj) {
if (this == obj) return true;
if (!(obj instanceof User other)) return false;
return Objects.equals(this.id, other.id);
}
}

这个类只重写了 equals(),没有重写 hashCode()。放进 HashSet 时,相同 id 的对象可能因为 hash 不同,被放到不同桶里,导致去重失败。

正确做法:

class User {
private Long id;
@Override
public boolean equals(Object obj) {
if (this == obj) return true;
if (!(obj instanceof User other)) return false;
return Objects.equals(this.id, other.id);
}
@Override
public int hashCode() {
return Objects.hash(id);
}
}

面试里可以总结成一句话:

只要重写 equals(),通常就必须重写 hashCode(),否则哈希集合和哈希映射可能出现行为异常。

HashSet 的底层主要依赖 HashMap

可以粗略理解成:

HashSet<E>
内部维护 HashMap<E, Object>

HashSet 添加元素时,本质上是把元素作为 HashMap 的 key 存进去:

map.put(element, PRESENT);

因为 HashMap 的 key 不能重复,所以 HashSet 就天然具备了去重能力。

因此,HashSet 的几个特点可以这样理解:

  • 不允许重复元素。
  • 不保证插入顺序。
  • 底层依赖 hashCode()equals() 判断重复。
  • 查询、插入、删除在理想情况下接近 O(1)

如果需要保持插入顺序,可以使用 LinkedHashSet。如果需要排序,可以使用 TreeSet

十一、二叉树、二叉搜索树和平衡二叉树

Section titled “十一、二叉树、二叉搜索树和平衡二叉树”

普通二叉树只限制每个节点最多有两个子节点,本身不要求元素大小关系。

二叉搜索树、退化树和四种平衡旋转示意图

二叉搜索树则有明确规则:

左子树节点 < 当前节点 < 右子树节点

例如下面这棵树就是一棵二叉搜索树:

8
/ \
4 12
/ \ / \
2 6 10 14

每个节点都满足:

左边比自己小,右边比自己大

插入时:

  • 比当前节点小,放左边。
  • 比当前节点大,放右边。
  • 如果不允许重复,相等元素不再插入。

二叉搜索树的中序遍历可以得到升序结果:

左 -> 根 -> 右

以上面这棵树为例,中序遍历结果是:

2 -> 4 -> 6 -> 8 -> 10 -> 12 -> 14

但普通二叉搜索树有一个问题:如果插入数据本身接近有序,就可能退化成链表。

例如连续插入:

1, 2, 3, 4, 5

树可能长成这样:

1
\
2
\
3
\
4
\
5

这时查找复杂度会从 O(log n) 退化到 O(n)

平衡二叉树就是为了解决这个问题。它要求任意节点左右子树高度差不能过大,常见规则是高度差不超过 1。插入或删除节点后,如果树失衡,就通过旋转恢复平衡。

常见失衡情况可以先记四种:

  • 左左:一次右旋。
  • 左右:先局部左旋,再整体右旋。
  • 右右:一次左旋。
  • 右左:先局部右旋,再整体左旋。

左左失衡,做一次右旋:

旋转前:
30
/
20
/
10
右旋后:
20
/ \
10 30

右右失衡,做一次左旋:

旋转前:
10
\
20
\
30
左旋后:
20
/ \
10 30

左右失衡,先对左子树左旋,再对整体右旋:

旋转前:
30
/
10
\
20
先局部左旋:
30
/
20
/
10
再整体右旋:
20
/ \
10 30

右左失衡,先对右子树右旋,再对整体左旋:

旋转前:
10
\
30
/
20
先局部右旋:
10
\
20
\
30
再整体左旋:
20
/ \
10 30

这几张图可以帮助记忆:哪边重,就先看它是不是同方向;同方向一次旋转,折线方向两次旋转。

红黑树是一种自平衡二叉搜索树。

它不像严格平衡二叉树那样要求左右高度差非常精确,而是通过节点颜色和一组规则,让树保持“大致平衡”。

可以简单理解为:

红黑树 = 带颜色规则的二叉搜索树

它的目标是避免树退化成链表,使查找、插入、删除都能保持在 O(log n) 级别。

Java 中:

  • TreeMap 底层是红黑树。
  • TreeSet 底层通常基于 TreeMap
  • HashMap 在桶内链表过长时,也可能把链表树化成红黑树。

TreeSet 的特点:

  • 元素不能重复。
  • 不保留插入顺序。
  • 会按照自然顺序或自定义比较器排序。
  • 线程不安全。

示例:

Set<Integer> set = new TreeSet<>();
set.add(3);
set.add(1);
set.add(2);
System.out.println(set); // [1, 2, 3]

如果放入自定义对象,需要让对象实现 Comparable,或者创建 TreeSet 时传入 Comparator

如果面试官问 HashMap 原理,可以这样答:

HashMap 底层是数组加链表或红黑树。put 时先计算 key 的 hash,再通过数组长度计算桶下标。如果没有冲突直接放入;如果有冲突,就在桶内链表或红黑树中比较 key。默认初始容量是 16,负载因子是 0.75,超过阈值会扩容。Java 8 之后,当桶内链表过长并且数组容量满足条件时,链表会树化成红黑树,以优化极端冲突下的查询性能。

如果问 HashSet 原理,可以这样答:

HashSet 底层依赖 HashMap,元素作为 HashMap 的 key 保存,value 使用一个固定占位对象。因此 HashSet 不允许重复元素,去重依赖 hashCode()equals()

如果问 ListSet 区别,可以这样答:

List 有序、可重复、可按下标访问;Set 不允许重复,通常不能按下标访问。HashSet 不保证顺序,LinkedHashSet 保持插入顺序,TreeSet 按比较规则排序。

如果问 TreeSet 原理,可以这样答:

TreeSet 底层基于红黑树,元素不能重复,并按自然顺序或比较器排序。它不保留插入顺序,查找、插入、删除通常是 O(log n)

这几个知识点可以串成一张图:

Collection
├── List
│ ├── ArrayList
│ └── LinkedList
└── Set
├── HashSet -> HashMap -> 哈希表
├── LinkedHashSet -> HashMap + 链表顺序
└── TreeSet -> TreeMap -> 红黑树
Map
├── HashMap -> 数组 + 链表 + 红黑树
└── TreeMap -> 红黑树

复习时不要只背结论,而要抓住底层结构:

  • List 关注顺序和下标。
  • Set 关注唯一性。
  • HashMap 关注哈希、冲突、扩容和树化。
  • HashSet 关注 HashMap 的 key 去重。
  • TreeSetTreeMap 关注红黑树排序。

这样从使用层一路讲到底层结构,面试回答会更完整。

队列、栈与 ArrayDeque 面试题整理

队列和栈是数据结构里最基础、也最容易在面试中被追问的两个概念。

它们看起来都像是“存放一组元素的容器”,但核心区别在于元素进出顺序操作位置限制。如果再放到 Java 集合体系里,还会涉及 QueueDequeListStackArrayDeque 这些类和接口的关系。

这篇文章按面试回答的方式整理一遍。

队列遵循 先进先出

Queue: First In First Out, FIFO

也就是说,先进入队列的元素,会先被取出。

入队顺序:A -> B -> C
出队顺序:A -> B -> C

栈遵循 先进后出

Stack: First In Last Out, FILO

也可以说是 后进先出

Last In First Out, LIFO

也就是说,最后进入栈的元素,会最先被取出。

入栈顺序:A -> B -> C
出栈顺序:C -> B -> A

这是队列和栈最核心的区别。

队列的插入和删除发生在不同端:

  • 在一端插入元素,通常称为入队。
  • 在另一端删除元素,通常称为出队。

可以理解成排队买票:

队尾入队 -> [A, B, C] -> 队头出队

栈的插入和删除发生在同一端:

  • 插入元素叫入栈。
  • 删除元素叫出栈。
  • 能直接操作的一端叫栈顶。

可以理解成往桌上叠盘子:

栈顶
C
B
A
栈底

新盘子放在最上面,拿的时候也只能从最上面拿。

队列通常适合按顺序处理任务,例如消息队列、线程池任务队列、广度优先搜索等场景。

队列在逻辑上更强调“从头部取、从尾部放”。如果底层是链表结构,可以通过节点指针向后遍历;如果底层是数组结构,也可以通过下标或循环数组逻辑访问。

栈更强调“只看栈顶”。如果要取出最早进入栈底的元素,就必须先把上面的元素依次弹出。

例如:

栈顶 -> C
B
栈底 -> A

如果想拿到 A,就要先处理 CB

需要注意的是,面试里有时会说“队列遍历比栈快”。这个说法不能绝对化,因为遍历速度还取决于底层实现:

  • ArrayDeque 基于数组实现,访问和扩容方式与链表不同。
  • LinkedList 基于链表实现,遍历依赖节点引用。
  • Stack 继承自 Vector,底层也是数组结构,并且很多方法带同步开销。

所以更准确的说法是:队列和栈限制的是操作语义,不是简单决定遍历速度。具体性能要看底层实现。

从 Java 集合体系看,队列和栈并不是完全对称的两个接口。

队列通常使用 Queue 接口:

Queue<Integer> queue = new ArrayDeque<>();

Queue 继承自 Collection,常见方法包括:

offer(e) // 入队,失败时通常返回 false
poll() // 出队,队列为空时返回 null
peek() // 查看队头,队列为空时返回 null

栈在早期可以使用 Stack 类:

Stack<Integer> stack = new Stack<>();

但在现代 Java 代码里,通常更推荐用 Deque 来模拟栈:

Deque<Integer> stack = new ArrayDeque<>();

常见方法包括:

push(e) // 入栈
pop() // 出栈
peek() // 查看栈顶

原因是 Stack 是一个比较旧的类,它继承自 VectorVector 的很多方法带同步机制,在大多数普通场景下并不是首选。

因此,面试里可以这样回答:

Java 中队列通常使用 QueueDeque;栈不一定要用 Stack 类,实际开发中更推荐使用 Deque,常见实现是 ArrayDeque

ArrayDequeDeque 接口的一个常用实现。

Deque 的全称是 double-ended queue,也就是双端队列。它允许在两端插入和删除元素,因此既可以当队列用,也可以当栈用。

当作队列:

Deque<String> queue = new ArrayDeque<>();
queue.offer("A");
queue.offer("B");
queue.offer("C");
System.out.println(queue.poll()); // A
System.out.println(queue.poll()); // B
System.out.println(queue.poll()); // C

当作栈:

Deque<String> stack = new ArrayDeque<>();
stack.push("A");
stack.push("B");
stack.push("C");
System.out.println(stack.pop()); // C
System.out.println(stack.pop()); // B
System.out.println(stack.pop()); // A

可以看到,同样是 ArrayDeque,只要使用的方法不同,就可以表达不同的数据结构语义。

如果面试官问“队列和栈有什么区别”,可以按这几个点回答:

  1. 规则不同:队列是 FIFO,栈是 FILO/LIFO。
  2. 操作位置不同:队列一端插入、另一端删除;栈只在栈顶插入和删除。
  3. 使用场景不同:队列适合按顺序处理任务;栈适合回退、括号匹配、函数调用、深度优先搜索等场景。
  4. Java 实现不同:队列常用 QueueDeque;栈可以用 Stack,但更推荐用 Deque 的实现类 ArrayDeque
  5. 性能不能只按“队列快、栈慢”简单判断,要结合底层结构,例如数组、链表、同步开销和扩容机制。

队列和栈的区别可以先抓住一句话:

队列:先进先出,像排队。
栈:先进后出,像叠盘子。

在 Java 中,还要补充一句:

需要队列或栈语义时,优先考虑 Queue、Deque 和 ArrayDeque;不要一看到栈就只想到 Stack。

这样回答既覆盖了基础概念,也能体现出对 Java 集合体系的理解。

React 父子组件通信面试题复盘

这是我曾经遇到过的一道 React 面试题,问题很常见:

React 父子组件之间如何通信?

这个问题看起来很基础,但其实很适合继续追问。因为它背后不只是 props 怎么传,而是 React 的数据流、状态放在哪里、组件边界怎么设计、什么时候用 Context、什么时候才需要状态管理库。

如果只是回答“父传子用 props,子传父用回调函数”,当然没错,但面试里还不够。更好的回答应该能顺着这个问题,把 React 的单向数据流讲清楚。

React 默认是单向数据流。

父组件可以通过 props 把数据传给子组件;子组件不能直接修改父组件的数据,如果子组件需要影响父组件,就由父组件传一个函数给子组件,子组件调用这个函数,把变化通知给父组件。

可以先用一句话概括:

父传子用 props,子传父用回调函数;如果兄弟组件之间要通信,就把状态提升到它们共同的父组件。

这是这道题的核心答案。

父传子最简单,就是把数据作为 props 传给子组件。

Parent.tsx
type User = {
id: number;
name: string;
};
function Parent() {
const user: User = {
id: 1,
name: 'xiaoxi',
};
return <UserCard user={user} />;
}
type UserCardProps = {
user: User;
};
function UserCard({ user }: UserCardProps) {
return (
<section>
<h2>{user.name}</h2>
<p>ID: {user.id}</p>
</section>
);
}

这里的数据方向很清楚:

Parent state/data -> props -> Child render

子组件只负责接收和展示,不负责决定这个数据从哪里来。

子组件不能直接修改父组件内部的 state。如果子组件需要触发父组件更新,就由父组件把更新函数传下去。

Counter.tsx
import { useState } from 'react';
function Parent() {
const [count, setCount] = useState(0);
const handleAdd = () => {
setCount((value) => value + 1);
};
return (
<div>
<p>当前数量:{count}</p>
<CounterButton onAdd={handleAdd} />
</div>
);
}
type CounterButtonProps = {
onAdd: () => void;
};
function CounterButton({ onAdd }: CounterButtonProps) {
return <button onClick={onAdd}>加一</button>;
}

这个过程可以理解为:

Child click -> call props function -> Parent setState -> Child receives new props

也就是说,子组件并没有“改父组件”,它只是触发了父组件提供的回调。

有时候子组件不只是触发动作,还需要把自己的数据交给父组件。

比如搜索框输入内容,父组件需要拿到关键词:

SearchBox.tsx
import { useState } from 'react';
function Parent() {
const [keyword, setKeyword] = useState('');
return (
<div>
<SearchBox onSearch={setKeyword} />
<p>当前搜索词:{keyword}</p>
</div>
);
}
type SearchBoxProps = {
onSearch: (keyword: string) => void;
};
function SearchBox({ onSearch }: SearchBoxProps) {
const [value, setValue] = useState('');
const handleSubmit = () => {
onSearch(value);
};
return (
<div>
<input value={value} onChange={(event) => setValue(event.target.value)} />
<button onClick={handleSubmit}>搜索</button>
</div>
);
}

这里的 onSearch(value) 就是典型的子传父。

面试时可以强调:React 里子传父不是反向修改数据,而是调用父组件传入的回调。

如果两个兄弟组件需要共享数据,通常不是让它们互相调用,而是把状态提升到共同父组件。

StateLifting.tsx
import { useState } from 'react';
function Parent() {
const [selectedId, setSelectedId] = useState<number | null>(null);
return (
<div>
<ProductList onSelect={setSelectedId} />
<ProductDetail productId={selectedId} />
</div>
);
}
type ProductListProps = {
onSelect: (id: number) => void;
};
function ProductList({ onSelect }: ProductListProps) {
return (
<ul>
<li onClick={() => onSelect(1)}>商品 1</li>
<li onClick={() => onSelect(2)}>商品 2</li>
</ul>
);
}
type ProductDetailProps = {
productId: number | null;
};
function ProductDetail({ productId }: ProductDetailProps) {
if (productId === null) return <p>请选择一个商品</p>;
return <p>当前商品 ID:{productId}</p>;
}

这就是 React 组件设计里很重要的一点:状态应该放在需要它的最小公共父组件中。

如果状态只属于一个组件,就放在这个组件内部;如果多个子组件都要用,就提升到公共父组件;如果很多层都要用,再考虑 Context 或状态管理库。

表单场景里,受控组件其实也是父子通信的一种体现。

父组件控制输入框的值,子组件只负责展示和触发修改。

ControlledInput.tsx
import { useState } from 'react';
function Parent() {
const [email, setEmail] = useState('');
return <EmailInput value={email} onChange={setEmail} />;
}
type EmailInputProps = {
value: string;
onChange: (value: string) => void;
};
function EmailInput({ value, onChange }: EmailInputProps) {
return (
<input
value={value}
onChange={(event) => onChange(event.target.value)}
placeholder="请输入邮箱"
/>
);
}

这里的核心是:

  • value 从父组件传给子组件。
  • onChange 从父组件传给子组件。
  • 子组件触发 onChange,父组件更新状态。
  • 新状态再通过 value 传回来。

所以受控组件不是“输入框自己管理值”,而是父组件管理值。

如果只是父子组件通信,不需要上来就用 Context。

但如果数据需要跨很多层传递,比如主题、登录用户、语言配置,就可以考虑 Context。

ThemeContext.tsx
import { createContext, useContext } from 'react';
type Theme = 'light' | 'dark';
const ThemeContext = createContext<Theme>('light');
function App() {
return (
<ThemeContext.Provider value="dark">
<Layout />
</ThemeContext.Provider>
);
}
function Layout() {
return <Toolbar />;
}
function Toolbar() {
const theme = useContext(ThemeContext);
return <button className={theme}>保存</button>;
}

Context 解决的是 props drilling,也就是一层层传 props 的问题。

但是它不应该被滥用。不是所有父子通信都需要 Context。对于很近的父子组件,直接用 props 更清楚。

有些场景不是传数据,而是父组件需要调用子组件暴露出来的方法。

比如父组件点击按钮,让子组件内部的输入框聚焦。

这时可以使用 forwardRefuseImperativeHandle

FocusInput.tsx
import { forwardRef, useImperativeHandle, useRef } from 'react';
type FocusInputRef = {
focus: () => void;
};
const FocusInput = forwardRef<FocusInputRef>(function FocusInput(_, ref) {
const inputRef = useRef<HTMLInputElement>(null);
useImperativeHandle(ref, () => ({
focus() {
inputRef.current?.focus();
},
}));
return <input ref={inputRef} placeholder="请输入内容" />;
});
function Parent() {
const inputRef = useRef<FocusInputRef>(null);
return (
<div>
<FocusInput ref={inputRef} />
<button onClick={() => inputRef.current?.focus()}>聚焦输入框</button>
</div>
);
}

不过这种方式要谨慎使用。

React 更推荐声明式的数据流。ref 更适合处理 DOM、聚焦、滚动、播放控制这类命令式场景,不适合作为普通业务数据通信的首选方案。

性能相关:回调函数会不会导致重复渲染

Section titled “性能相关:回调函数会不会导致重复渲染”

面试官可能继续问:父组件每次渲染都会创建新的回调函数,会不会导致子组件重复渲染?

答案是:可能会,但要结合场景看。

如果子组件使用了 memo,并且传入的回调函数每次都是新引用,子组件可能仍然会重新渲染。这时可以使用 useCallback 稳定函数引用。

MemoCallback.tsx
import { memo, useCallback, useState } from 'react';
const Child = memo(function Child({ onAdd }: { onAdd: () => void }) {
return <button onClick={onAdd}>加一</button>;
});
function Parent() {
const [count, setCount] = useState(0);
const handleAdd = useCallback(() => {
setCount((value) => value + 1);
}, []);
return (
<div>
<p>{count}</p>
<Child onAdd={handleAdd} />
</div>
);
}

但不要为了“看起来专业”到处写 useCallback。如果子组件没有性能问题,或者没有使用 memo,盲目使用 useCallback 反而会增加理解成本。

父子通信、兄弟通信、跨层级通信,本质上都是状态在哪里的问题。

可以按下面的顺序判断:

场景推荐方式
父组件给子组件数据props
子组件通知父组件回调函数
兄弟组件共享状态状态提升
多层组件共享稳定数据Context
大量页面共享复杂业务状态Zustand、Redux、Jotai 等状态管理库

状态管理库不是为了替代 props,而是为了解决更大范围、更复杂的数据共享和更新问题。

比如用户信息、购物车、权限、全局弹窗、复杂筛选条件,这类状态可能会跨多个页面或模块使用,就可以考虑状态管理库。

这道题可以这样回答:

React 是单向数据流。父组件向子组件传值用 props;子组件想影响父组件时,父组件传回调函数给子组件,子组件调用回调并把数据传回去。如果兄弟组件需要共享状态,就把状态提升到共同父组件。如果跨层级传递太深,可以用 Context;如果是跨页面、跨模块的复杂共享状态,再考虑 Redux、Zustand 这类状态管理库。特殊情况下,父组件需要调用子组件内部方法,可以用 refforwardRefuseImperativeHandle,但这更适合聚焦、滚动这类命令式场景,不是普通数据通信的首选。

如果面试官继续追问,可以展开这些点:

  • React 为什么强调单向数据流?
  • 状态提升解决什么问题?
  • Context 和状态管理库有什么区别?
  • 受控组件为什么也是父子通信?
  • ref 能不能替代 props?
  • useCallback 是否一定能优化性能?

React 父子通信这道题看似基础,但它可以引申到很多 React 核心思想:

  • 数据从父组件流向子组件。
  • 子组件通过回调通知父组件。
  • 多个组件共享状态时,优先状态提升。
  • 跨层级共享再考虑 Context。
  • 复杂全局状态再考虑状态管理库。
  • ref 是命令式能力,不是普通数据流的替代品。

所以这道题最重要的不是记住几个 API,而是理解 React 组件之间的数据边界:谁拥有状态,谁负责修改状态,谁只是接收状态并渲染。

Redis 缓存穿透、击穿、雪崩面试题复盘

这是 Redis 面试里非常高频的一类问题:

什么是缓存穿透、缓存击穿和缓存雪崩?它们有什么区别?应该怎么解决?

这三个概念很容易混在一起,因为它们的结果都可能是“请求打到数据库,数据库压力变大”。但它们的根因完全不同。

可以先用一句话区分:

穿透:查的数据根本不存在。
击穿:一个热点 Key 过期。
雪崩:大量 Key 同时过期,或者 Redis 整体不可用。

缓存穿透指的是:请求的数据在缓存中不存在,在数据库中也不存在。

比如:

  • 请求一个不存在的用户 ID:id = -1
  • 请求一个已经删除的商品
  • 恶意攻击者构造大量非法参数

正常缓存流程一般是:

请求 -> 查 Redis -> Redis 没有 -> 查数据库 -> 写入 Redis -> 返回数据

但如果数据根本不存在,就会变成:

请求 -> 查 Redis -> Redis 没有 -> 查数据库 -> 数据库也没有 -> 返回空

问题在于:下一次同样的非法请求过来,Redis 里还是没有,于是又会打到数据库。

如果有人构造大量不存在的 ID,请求就会绕过缓存,持续打到数据库。

缓存穿透常见有两种解决方案:缓存空对象和布隆过滤器。

当数据库查不到数据时,也往 Redis 里写一个空值或特殊值,并设置较短过期时间。

cache-null.js
async function getUser(id) {
const cacheKey = `user:${id}`;
const cached = await redis.get(cacheKey);
if (cached !== null) {
return cached === 'NULL' ? null : JSON.parse(cached);
}
const user = await db.queryUserById(id);
if (!user) {
await redis.set(cacheKey, 'NULL', 'EX', 60);
return null;
}
await redis.set(cacheKey, JSON.stringify(user), 'EX', 3600);
return user;
}

这样下一次请求同一个不存在的 ID,就会被 Redis 拦住,不会继续打数据库。

它的优点是简单,缺点是如果恶意请求的非法 ID 极多,Redis 里会出现很多空值缓存。所以空值缓存的过期时间一般要短一些。

布隆过滤器适合在请求进入缓存和数据库之前,先判断这个数据是否可能存在。

请求 userId
|
v
布隆过滤器判断
|
| 不存在 -> 直接拒绝
|
| 可能存在
v
查 Redis / DB

布隆过滤器的特点是:

  • 如果它判断不存在,那就一定不存在。
  • 如果它判断存在,只能说明可能存在。

所以它适合挡掉大量明显非法的请求。

面试里可以这样说:缓存空对象适合兜住少量不存在数据,布隆过滤器适合在入口处拦截大量非法 Key。

缓存击穿指的是:某一个热点 Key 在过期瞬间,大量并发请求同时打到数据库。

注意,击穿通常是单个 Key。

比如:

  • 某个大促商品详情页
  • 某个热门直播间信息
  • 某个高访问量用户主页
  • 某条热点新闻

这个 Key 平时在 Redis 里,所以数据库压力不大。

但它刚好过期时,大量请求同时进来,发现 Redis 没有,于是一起查数据库。

热点 Key 过期
|
大量请求同时进入
|
Redis 都没查到
|
全部打到数据库

这就是缓存击穿。

缓存击穿常见有两种方案:互斥锁和逻辑过期。

互斥锁的思路是:同一时间只允许一个线程去查数据库并重建缓存,其他线程等待或重试。

mutex-lock.js
async function getProduct(id) {
const cacheKey = `product:${id}`;
const lockKey = `lock:product:${id}`;
const cached = await redis.get(cacheKey);
if (cached) return JSON.parse(cached);
const locked = await redis.set(lockKey, '1', 'NX', 'EX', 10);
if (!locked) {
await sleep(50);
return getProduct(id);
}
try {
const product = await db.queryProductById(id);
await redis.set(cacheKey, JSON.stringify(product), 'EX', 3600);
return product;
} finally {
await redis.del(lockKey);
}
}

这个方案可以保护数据库,但会让部分请求等待,接口耗时可能变长。

逻辑过期的思路是:Redis 里的数据不设置物理过期,或者设置很长的物理过期;真正的过期时间写在 value 里。

logical-expire-value.json
{
"data": {
"id": 1,
"name": "热门商品"
},
"expireAt": "2026-05-20T20:00:00+08:00"
}

请求到来时:

  • 如果逻辑时间没过期,直接返回数据。
  • 如果逻辑时间过期,当前线程尝试拿锁。
  • 拿到锁的线程异步重建缓存。
  • 当前请求先返回旧数据。
查到缓存
|
判断逻辑过期
|
已过期 -> 尝试加锁 -> 异步重建缓存
|
直接返回旧数据

它的优点是响应速度更稳,不会让大量请求等待数据库。

缺点是用户可能短时间看到旧数据,所以它更适合对实时性要求不那么高的热点数据。

缓存雪崩指的是:大量 Key 在同一时间过期,或者 Redis 服务不可用,导致大量请求同时打到数据库。

它和击穿的区别是:

  • 击穿是单个热点 Key。
  • 雪崩是大量 Key,或者缓存层整体出问题。

比如:

  • 批量导入缓存时设置了相同过期时间。
  • 活动开始前预热了一批商品缓存,过期时间完全一样。
  • Redis 宕机或网络故障。
  • Redis 集群大面积不可用。

雪崩的危害比击穿更大,因为它不是一个点,而是一片。

缓存雪崩通常要从多个层面解决。

不要让大量 Key 设置完全一样的过期时间。

random-expire.js
const baseTtl = 3600;
const randomTtl = Math.floor(Math.random() * 300);
await redis.set(cacheKey, value, 'EX', baseTtl + randomTtl);

这样可以把过期时间打散,避免某一秒大量 Key 同时失效。

可以在 Redis 前面加本地缓存,比如 Caffeine、Guava Cache 或进程内缓存。

请求 -> 本地缓存 -> Redis -> 数据库

即使 Redis 短时间抖动,本地缓存也能顶住一部分热点请求。

不过本地缓存也会带来一致性问题,所以一般只适合热点数据或允许短暂不一致的数据。

当数据库压力过大时,不能让所有请求继续堆积。

可以做:

  • 数据库访问限流。
  • 熔断降级。
  • 返回兜底数据。
  • 返回“服务器繁忙,请稍后再试”。

降级不是偷懒,而是在系统压力过大时保护核心链路。

如果雪崩原因是 Redis 宕机,就需要高可用架构。

常见方案:

  • Redis 主从复制。
  • 哨兵模式。
  • Redis Cluster。
  • 多机房或多可用区部署。

高可用解决的是缓存服务不可用的问题,过期时间随机解决的是大量 Key 同时失效的问题。两者关注点不同。

维度缓存穿透缓存击穿缓存雪崩
核心原因数据根本不存在热点 Key 过期大量 Key 同时过期或 Redis 宕机
Key 数量大量不存在 Key单个热点 Key多个或大部分 Key
数据库状态数据库里也没有数据库里有数据数据库里有数据
主要危害绕过缓存打数据库单个热点打爆数据库大量请求压垮数据库
主要方案缓存空对象、布隆过滤器互斥锁、逻辑过期过期随机、多级缓存、限流降级、高可用

记忆时可以抓住关键词:

穿透:不存在
击穿:热点
雪崩:大量

这道题可以这样回答:

缓存穿透是请求的数据在缓存和数据库中都不存在,请求会绕过缓存持续打到数据库,常用缓存空对象和布隆过滤器解决。缓存击穿是某个热点 Key 在过期瞬间被大量并发请求访问,导致请求同时打到数据库,常用互斥锁或逻辑过期解决。缓存雪崩是大量 Key 同时过期,或者 Redis 服务不可用,导致大量请求同时涌向数据库,常用过期时间加随机值、多级缓存、限流降级和 Redis 高可用解决。

如果面试官继续追问,可以补充方案取舍:

  • 缓存空对象简单,但要设置较短 TTL,避免缓存污染。
  • 布隆过滤器适合拦截大量非法请求,但存在误判为可能存在。
  • 互斥锁能保护数据库,但会增加等待时间。
  • 逻辑过期响应快,但可能返回旧数据。
  • 过期随机能打散失效时间,但不能解决 Redis 宕机。
  • Redis 高可用能提升可用性,但不能替代限流和降级。

这类 Redis 面试题的关键,是把三个概念的根因讲清楚。

缓存穿透:缓存没有,数据库也没有。
缓存击穿:缓存没有,但数据库有,而且是热点 Key。
缓存雪崩:大量缓存同时没有,或者 Redis 整体不可用。

只要先分清根因,再说解决方案,就不会混乱。

最后再补一句工程思维:缓存是为了保护数据库,但缓存系统本身也会失效。所以真正可靠的设计,通常不是只靠一个方案,而是缓存策略、限流降级、异步重建和高可用架构一起配合。