现身说法:2023 中级程序员进阶之路
关于程序员的几个阶段
每个程序员、或者说每个工作者都应该有自己的职业规划,如果看到这里的朋友没有自己的职业规划,希望你可以思考一下自己的将来。
LZ 常常思考自己的未来,也从自己的思考中总结出了一些东西,作为第一部分来谈谈。LZ 认为一名程序员应该有几个阶段(以下时间都算上实习期):
第一阶段----三年
我认为三年对于程序员来说是第一个门槛,这个阶段将会淘汰掉一批不适合写代码的人。这一阶段,我们走出校园,迈入社会,成为一名程序员,正式从书本上的内容迈向真正的企业级开发。我们知道如何团队协作、如何使用项目管理工具、项目版本如何控制、我们写的代码如何测试如何在线上运行等等,积累了一定的开发经验,也对代码有了一定深入的认识,是一个比较纯粹的 Coder 的阶段
第二阶段----五年第三阶段----十年
五年又是区分程序员的第二个门槛。有些人在三年里,除了完成工作,在空余时间基本不会研究别的东西,这些人永远就是个 Coder,年纪大一些势必被更年轻的人给顶替;有些人在三年里,除了写代码之外,还热衷于研究各种技术实现细节、看了 N 多好书、写一些博客、在 Github 上分享技术,这些人在五年后必然具备在技术上独当一面的能力并且清楚自己未来的发展方向,从一个 Coder 逐步走向系统分析师或是架构师,成为项目组中不可或缺的人物
十年又是另一个门槛了,转行或是继续做一名程序员就在这个节点上。如果在前几年就抱定不转行的思路并且为之努力的话,那么在十年的这个节点上,有些人必然成长为一名对行业有着深入认识、对技术有着深入认识、能从零开始对一个产品进行分析的程序员,这样的人在公司基本担任的都是 CTO、技术专家、首席架构师等最关键的职位,这对于自己绝对是一件荣耀的事,当然老板在经济上也绝不会亏待你
第一部分总结一下,我认为,随着你工作年限的增长、对生活对生命认识的深入,应当不断思考三个问题:
我到底适不适合当一名程序员?
我到底应不应该一辈子以程序员为职业?
我对编程到底持有的是一种什么样的态度,是够用就好呢还是不断研究?
最终,明确自己的职业规划,对自己的规划负责并为之努力。
关于项目经验
LZ 在网上经常看到一些别的朋友有提出项目经验的问题,依照 LZ 面试的感觉来说,面试主要看几点:项目经验+基本技术+个人潜力(也就是值不值得培养)。
关于项目经验,我认为并发编程网的创始人方腾飞老师讲的一段话非常好:
复制代码
我觉得这就已经足以概括了。也许你仅仅工作一年,也许你做的是项目中微不足道的模块,当然这些一定是你的劣势且无法改变,但是如何弥补这个劣势,从方老师的话中我总结几点:
明确你的项目到底是做什么的,有哪些功能
明确你的项目的整体架构,在面试的时候能够清楚地画给面试官看并且清楚地指出从哪里调用到哪里、使用什么方式调用
明确你的模块在整个项目中所处的位置及作用
明确你的模块用到了哪些技术,更好一些的可以再了解一下整个项目用到了哪些技术
在你无法改变自己的工作年限、自己的不那么有说服力的项目经验的情况下(这一定是扣分项),可以通过这种方式来一定程度上地弥补并且增进面试官对你的好感度。
补充一点,在面试中聊你的项目的时候,有一个问题 90%是绕不过的:谈一下你在项目中解决过的比较复杂的问题。这需要在工作中不断去发现和探索,不需要多,在你自己目前的项目中只要你找到一两个能说的问题就行。一个小技巧是,即使问题不是你解决的而是别人解决的,但是你把这个问题弄懂、搞透了,在面试的时候你一样可以把这个问题当作是你自己解决的来说----毕竟,谁来管这个问题当时到底是不是你解决的呢?
关于专业技能
写完项目接着写写一名 3 年工作经验的 Java 程序员应该具备的技能,这可能是 Java 程序员们比较关心的内容。我这里要说明一下,以下列举的内容不是都要会的东西----但是如果你掌握得越多,最终能得到的评价、拿到的薪水势必也越高。
1、基本语法
这包括 static、final、transient 等关键字的作用,foreach 循环的原理等等。今天面试我问你 static 关键字有哪些作用,如果你答出 static 修饰变量、修饰方法我会认为你合格,答出静态块,我会认为你不错,答出静态内部类我会认为你很好,答出静态导包我会对你很满意,因为能看出你非常热衷研究技术。
最深入的一次,LZ 记得面试官直接问到了我 volatile 关键字的底层实现原理(顺便插一句,面试和被面试本身就是相对的,面试官能问这个问题同时也让面试者感觉到面试官也是一个喜爱研究技术的人,增加了面试者对公司的好感,LZ 最终选择的就是问了这个问题的公司),不要觉得这太吹毛求疵了----越简单的问题越能看出一个人的水平,别人对你技术的考量绝大多数都是以深度优先、广度次之为标准的,切记。
2、集合
非常重要,也是必问的内容。基本上就是 List、Map、Set,问的是各种实现类的底层实现原理,实现类的优缺点。
集合要掌握的是 ArrayList、LinkedList、Hashtable、HashMap、ConcurrentHashMap、HashSet 的实现原理,能流利作答,当然能掌握 CopyOnWrite 容器和 Queue 是再好不过的了。另外多说一句,ConcurrentHashMap 的问题在面试中问得特别多,大概是因为这个类可以衍生出非常多的问题,关于 ConcurrentHashMap,我给网友朋友们提供三点回答或者是研究方向:
(1)ConcurrentHashMap 的锁分段技术
(2)ConcurrentHashMap 的读是否要加锁,为什么
(3)ConcurrentHashMap 的迭代器是强一致性的迭代器还是弱一致性的迭代器
3、设计模式
本来以为蛮重要的一块内容,结果只在阿里巴巴 B2B 事业部面试的时候被问了一次,当时问的是装饰器模式。
当然咱们不能这么功利,为了面试而学习,设计模式在工作中还是非常重要、非常有用的,23 种设计模式中重点研究常用的十来种就可以了,面试中关于设计模式的问答主要是三个方向:
(1)你的项目中用到了哪些设计模式,如何使用
(2)知道常用设计模式的优缺点
(3)能画出常用设计模式的 UML 图
4、多线程
这也是必问的一块了。因为三年工作经验,所以基本上不会再问你怎么实现多线程了,会问得深入一些比如说 Thread 和 Runnable 的区别和联系、多次 start 一个线程会怎么样、线程有哪些状态。当然这只是最基本的,出乎意料地,几次面试几乎都被同时问到了一个问题,问法不尽相同,总结起来是这么一个意思:
复制代码
聪明的网友们对这个问题是否有答案呢?不难,java.util.concurrent 下就有现成的类可以使用。
另外,线程池也是比较常问的一块,常用的线程池有几种?这几种线程池之间有什么区别和联系?线程池的实现原理是怎么样的?实际一些的,会给你一些具体的场景,让你回答这种场景该使用什么样的线程池比较合适。
最后,虽然这次面试问得不多,但是多线程同步、锁这块也是重点。synchronized 和 ReentrantLock 的区别、synchronized 锁普通方法和锁静态方法、死锁的原理及排查方法等等,关于多线程,我在之前有些过文章总结过多线程的 40 个问题,可以参看40个Java多线程问题总结。
5、低代码
编程语言本身在不断进步,对于菜鸟开发者来说,需要较高的学习成本。但低代码平台天然就具备全栈开发能力,低代码程序员天然就是全栈程序员。
首先,低代码不会取代专业开发者,更像是专业开发者的好朋友,可以相辅相成。它不仅能大大缩短应用的构建和发布时间,可以让开发者事半功倍,还允许非技术出身的开发人员构建不太复杂的应用程序,让更专业的开发者专注更复杂、重要的应用程序。
JNPF 低代码开发平台,一款 PaaS 服务的开发平台,深度集成 java+.net 6 双技术引擎,采用主流前后分离框架、代码生成器、强大的工作流引擎,将企业管理系统中 80%的功能通过组件化、可视化、拖拽化的方式来完成。支持私有化部署,全源码交付。这边强烈建议玩软件的试试它,你会发现不一样的惊喜!
你可以试试看:https://www.jnpfsoft.com/?infoq
6、IO
再次补充 IO 的内容,之前忘了写了。
IO 分为 File IO 和 Socket IO,File IO 基本上是不会问的,问也问不出什么来,平时会用就好了,另外记得 File IO 都是阻塞 IO。
Socket IO 是比较重要的一块,要搞懂的是阻塞/非阻塞的区别、同步/异步的区别,借此理解阻塞 IO、非阻塞 IO、多路复用 IO、异步 IO 这四种 IO 模型,Socket IO 如何和这四种模型相关联。这是基本一些的,深入一些的话,就会问 NIO 的原理、NIO 属于哪种 IO 模型、NIO 的三大组成等等,这有些难,当时我也是研究了很久才搞懂 NIO。提一句,NIO 并不是严格意义上的非阻塞 IO 而应该属于多路复用 IO,面试回答的时候要注意这个细节,讲到 NIO 会阻塞在 Selector 的 select 方法上会增加面试官对你的好感。
如果用过 Netty,可能会问一些 Netty 的东西,毕竟这个框架基本属于当前最好的 NIO 框架了(Mina 其实也不错,不过总体来说还是比不上 Netty 的),大多数互联网公司也都在用 Netty。
7、JDK 源码
要想拿高工资,JDK 源码不可不读。上面的内容可能还和具体场景联系起来,JDK 源码就是实打实地看你平时是不是爱钻研了。LZ 面试过程中被问了不少 JDK 源码的问题,其中最***钻的一个问了 LZ,String 的 hashCode()方法是怎么实现的,幸好 LZ 平时 String 源代码看得多,答了个大概。JDK 源码其实没什么好总结的,纯粹看个人,总结一下比较重要的源码:
(1)List、Map、Set 实现类的源代码
(2)ReentrantLock、AQS 的源代码
(3)AtomicInteger 的实现原理,主要能说清楚 CAS 机制并且 AtomicInteger 是如何利用 CAS 机制实现的
(4)线程池的实现原理
(5)Object 类中的方法以及每个方法的作用
这些其实要求蛮高的,LZ 去年一整年基本把 JDK 中重要类的源代码研究了个遍,真的花费时间、花费精力,当然回头看,是值得的----不仅仅是为了应付面试。
8、框架
老生常谈,面试必问的东西。一般来说会问你一下你们项目中使用的框架,然后给你一些场景问你用框架怎么做,比如我想要在 Spring 初始化 bean 的时候做一些事情该怎么做、想要在 bean 销毁的时候做一些事情该怎么做、MyBatis 中 $和 #的区别等等,这些都比较实际了,平时积累得好、有多学习框架的使用细节自然都不成问题。
如果上面你的问题答得好,面试官往往会深入地问一些框架的实现原理。问得最多的就是 Spring AOP 的实现原理,当然这个很简单啦,两句话就搞定的的事儿,即使你不会准备一下就好了。LZ 遇到的最变态的是让 LZ 画一下 Spring 的 Bean 工厂实现的 UML 图,当然面对这样一个有深度的问题,LZ 是绝对答不出来的/(ㄒoㄒ)/~~
9、数据库
数据库十有八九也都会问到。一些基本的像 union 和 union all 的区别、left join、几种索引及其区别就不谈了,比较重要的就是数据库性能的优化,如果对于数据库的性能优化一窍不通,那么有时间,还是建议你在面试前花一两天专门把 SQL 基础和 SQL 优化的内容准备一下。
不过数据库倒是不用担心,一家公司往往有很多部门,如果你对数据库不熟悉而基本技术又非常好,九成都是会要你的,估计会先把你放到对数据库使用不是要求非常高的部门锻炼一下。
10、数据结构和算法分析
数据结构和算法分析,对于一名程序员来说,会比不会好而且在工作中绝对能派上用场。数组、链表是基础,栈和队列深入一些但也不难,树挺重要的,比较重要的树 AVL 树、红黑树,可以不了解它们的具体实现,但是要知道什么是二叉查找树、什么是平衡树,AVL 树和红黑树的区别。记得某次面试,某个面试官和我聊到了数据库的索引,他问我:
复制代码
LZ 答到用的 Hash 表吧,答错。他又问,你知道为什么要使用树吗?LZ 答到因为 Hash 表可能会出现比较多的冲突,在千万甚至是上亿级别的数据面前,会大大增加查找的时间复杂度。而树比较稳定,基本保证最多二三十次就能找到想要的数据,对方说不完全对,最后我们还是交流了一下这个问题,我也明白了为什么要使用树,这里不说,网友朋友们觉得索引为什么要使用树来实现呢?
至于算法分析,不会、不想研究就算了,记得某次面试对方问我,Collections.sort 方法使用的是哪种排序方法,额,吐血三升。当然为了显示 LZ 的博学,对算法分析也有一定的研究(⊙﹏⊙)b,LZ 还是硬着头皮说了一句可能是冒泡排序吧。当然答案肯定不是,有兴趣的网友朋友们可以去看一下 Collections.sort 方法的源代码,用的是一种叫做 TimSort 的排序法,也就是增强型的归并排序法。
11、Java 虚拟机
出乎 LZ 的意料,Java 虚拟机应该是很重要的一块内容,结果在这几家公司中被问到的概率几乎为 0。要知道,LZ 去年可是花了大量的时间去研究 Java 虚拟机的,光周志明老师的《深入理解 Java 虚拟机:JVM 高级特性与最佳实践》,LZ 就读了不下五遍。
言归正传,虽然 Java 虚拟机没问到,但我觉得还是有必要研究的,LZ 就简单地列一个提纲吧,谈谈 Java 虚拟机中比较重要的内容:
Java 虚拟机的内存布局
GC 算法及几种垃圾收集器
类加载机制,也就是双亲委派模型
Java 内存模型
happens-before 规则
volatile 关键字使用规则
也许面试无用,但在走向大牛的路上,不可不会。
最后,如果有兴趣有时间,建议学习、研究一下 SOA 和 RPC,面向服务体系,大型分布式架构必备,救命良方、包治百病、屡试不爽。
评论