5千和2万,普通程序员和熟练开发者的差别

本文探讨了从普通开发者成长为熟练开发者的路径。强调了构建知识体系的重要性,并提供了具体的技能水平要求作为参考。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在“开发者山行图:程序员层级划分”一文中,我们根据职场发展通路和能力水平,把程序员分成了如下四个等级:

  1. 普通开发者
  2. 熟练开发者、高级开发工程师、技术组长
  3. 技术专家、架构师、一线经理
  4. 科学家、首席(资深)架构师、部门研发总监

今天这篇文章,我们来讨论普通开发者如何进阶为熟练开发者。

我们会先定义普通开发者应该达到的能力水平,然后再介绍他应该在哪方面提升才可能进到下一个阶段。

普通开发者的能力水平

先看我经常提到的一张图——技术成长阶段图:

普通开发者处在第一个成长阶段:专项能力提升。

专项能力提升阶段是初级阶段,你为了搞定事情,必须先具备某些基础能力,比如某种编程语言(Java / C++ / Python / JavaScript 等)、某个IDE (Visual Studio / Qt Creator / Android Studio / Eclipse / WebStorm 等)、某种技术框架(Netty / Qt / SSH / MyBatis / AngularJS 等)。

这个阶段最重要的就是提升专项能力,让自己能够迅速搞定一些别人安排给你的事情,体现出你的价值。

在这个阶段,最起码达到下列水准:

  • 能理解并完成别人分解好的叶子任务
  • 能自己完成一个功能模块的开发

以前端开发者为例,在这个阶段,你必须能将从 UI 过来的某个页面原型转换为可运行、可操作的真实 Web 页面。比如登录页面、比如商品详情展示页面,要能做到实现可视化页面和业务逻辑跳转。

以 Android App 开发者为例,在这个阶段,你必须能搞定类似登录、支付等这类功能。以 App 的登录功能为例,你要能使用特定的布局和组件实现自适应的登录界面,还要能学习使用类似微信、微博等第三方账户的 SDK ,实现第三方账户登录。

你可能很难自己设计一套完整的登录功能(包含口令加密、单点登录、Cookie 使用、验证码、密码找回、第三方登录、多终端同步、角色鉴权、用户数据存储等),但是当熟练开发者或者高级开发者完成了设计,把某个小功能分配给你的时候,你要能够理解这个设计,并将这个设计实现出来,和整个登录系统整合在一起,能够正常运转

当你能够搞定这些事情的时候,你的专项技术水平起码是这样的:

  • 能熟练搭建需要的开发和测试环境
  • 熟悉所用编程语言的基础语法,有这门语言的知识地图,熟悉该语言本身的各种基础功能,了解这门语言的高级功能
  • 能调用已有技术框架的 API 来实现特定功能
  • 能在实现同一功能的多组 API 之间选择合适的那一组
  • 能熟练使用开发环境和各种辅助工具进行调试
  • 可以解决易复现的、逻辑简单的软件 Bug
  • 能自主学习某个新框架的基础 API ,参考 DEMO 完成某个功能的基本实现

我的总结可能不完善或存在 Bug ,请大家在留言区指正、补充。

我想到一个身边的例子,刚好和我们讨论的这个话题契合。

有一次和朋友 G 聊天,他说他部门有位做前端开发的程序员 X ,一直在用 jQuery 实现各种页面,代码冗余,结构凌乱。Vue.js 出来后,G 想在新版本中使用,就安排 X 学习 Vue.js ,让他参考公司产品某两个页面的功能和布局,简化设计,做两个带异步数据请求功能的页面出来看看效果, 结果一个星期过后,X 告诉 G ,Vue.js 太复杂,开发文档看不明白,代码也搞不懂,进行不下去……

如果你像 X 这样的状态,就说明你还停留在普通开发者阶段,只能在他人指导和安排下编码实现特定的细分功能。

要想进阶,就要先在专项技术能力上达到我们前面所列清单描述的水平(那是基础要求),然后再对照熟练开发者的能力要求,有针对性的去学习提高,稳步进入下个阶段。

普通和熟练的关键区别

对照前面的技术成长阶段图,熟练开发者处在中间阶段,拥有自己的技能体系。

为了搞明白拥有自己的技能体系是什么概念,我们先来看两组概念:

  1. 知识、知识体系
  2. 技能、技能体系

简单说,知识是能引发改变的信息,技能是运用知识解决问题的一系列思维和动作的组合。但这样讲不大容易明白,我们举个软件开发领域的例子。 C++ 中的虚函数是一个知识点,用虚函数这个特性定义模块接口,实现模块的动态加载,就是技能。

接下来为了行文方便,我不再区分知识和技能,直接把知识等同于技能,甚至用知识一词代替技能。请在阅读时根据上下文加以区分。

知识和知识体系的关系,技能和技能体系的关系,就像树木和森林的关系,就像瑞士军刀上小刀、小剪子、小螺丝刀和瑞士军刀的关系。

这样打个比方,应该可以体会到“体系”是什么了。

所谓知识体系,就是若干相互联系的、可以用来解决特定场景问题的知识的组合。

Ok,现在回头来看普通开发者和熟练开发者的区别,就简单多了。

普通开发者因为不断完成任务,不断做项目,接触、使用了各种各样的“知识点”,于是拥有了各种离散的知识,但这个知识和那个知识很少关联,他拥有的知识,是散兵游勇,如下图所示:

当你的知识点各不关联时,你在遇到问题时,就很难进行系统思考,就很难获得解决办法。

而熟练开发者,则(被动或主动)经历了知识点的梳理、整合过程,把零散的知识点围绕着某个应用场景联系起来,形成了知识体系。就像下图:

当你有了知识体系之后,遇到和你知识体系对应的场景相关的问题,就可以快速找到相关的知识,系统的分析问题,最终解决问题。

你看,是否拥有知识体系,导致了熟练开发者和普通开发者的一个非常重要的区别:熟练开发者能够独立负责一个模块或子系统的设计和开发工作,拥有分析问题、解决问题的能力和任务分解的能力,可以分配任务给普通开发者

进阶修炼方向

好啦,从技术水平的角度讲,普通开发者和熟练开发者的区别就在于:普通开发者的知识不成体系,而熟练开发者构建了与某个应用场景相关的知识体系。

这样看来,普通开发者要想进阶,方向就非常明确了:构建自己的知识体系。

那么,怎么构建知识体系呢?这需要很长的篇幅来讲明白,下回见。先参考这篇文章:构建知识图谱,让自己更值钱。如果后续有机会,我会重构一篇新的文章,再聊聊构建知识体系这个话题。


相关阅读:

<think>我们遇到了一个编译错误。在Java中,数组初始化器(即使用大括号{}来初始化数组)只能在声明变量时使用,或者在创建数组对象时使用(通过new关键字)。在赋值语句中,不能直接使用数组初始化器,除非同时创建新的数组对象。观察代码:第一行:String[]names={"板","桃花心","杉木"};//这是正确的,因为在声明的同时初始化第二行:names={"板","桃花心","杉木"};//这是错误的,因为这里没有使用new关键字,而且不是在声明的时候。所以,第二行会出现“Arrayinitializerisnotallowedhere”的错误。修正方法:在赋值时,使用new关键字并指定数组类型,或者使用匿名数组的方式。以下是修正后的代码:第一种方式(推荐):names=newString[]{"板","桃花心","杉木"};第二种方式,如果你想要一个全新的数组(比如想要一个不可修改的数组),可以使用:names=newString[]{"板","桃花心","杉木"};注意:第二种方式实际上第一种是一样的,都是创建了一个新的String数组。另一种方式是使用数组变量指向一个已经存在的数组,但这里我们是要重新赋值一个新数组。我们还可以使用Arrays类的copyOf方法,但这里直接赋值新的数组更直接。因此,将错误的行改为:names=newString[]{"板","桃花心","杉木"};下面给出完整的例子:</think> 在Java中,当直接赋值`{}`数组初始化器时会出现"Array initializer is not allowed here"错误。以下是错误代码分析及修正方案: ### 问题分析 ```java String[] names = {"板","桃花心","杉木"}; // ✅ 正确(声明时初始化) names = {"板","桃花心","杉木"}; // ❌ 错误(赋值时直接使用{}) ``` ### 错误原因 1. **语法规则限制**:Java允许在变量声明时使用`{}`初始化器,但不允许在常规赋值语句中直接使用 2. **类型推断缺失**:赋值操作需要显式指定数组类型,编译器无法自动推断`{}`的类型 ### 解决方案 ```java // 正确修正方案1:显式创建新数组 names = new String[]{"板", "桃花心", "杉木"}; // 正确修正方案2:通过临时变量赋值 String[] temp = {"板", "桃花心", "杉木"}; names = temp; // 正确修正方案3:使用Arrays工具类(Java 5+) import java.util.Arrays; names = Arrays.asList("板", "桃花心", "杉木").toArray(new String[0]); // 正确修正方案4:使用集合初始化(Java 9+) import java.util.List; names = List.of("板", "桃花心", "杉木").toArray(new String[0]); ``` ### 🧪 相关扩展:匿名数组使用场景 ```java // 场景1:方法参数传递 processNames(new String[]{"板","桃花心","杉木"}); // 场景2:返回值 public String[] getWoodNames() { return new String[]{"板","桃花心","杉木"}; // ✅ // return {"板","桃花心","杉木"}; // ❌ } ``` ### ✅ 最佳实践 1. 声明时初始化使用`Type[] var = {元素...};`语法 2. 重新赋值必须使用`var = new Type[]{元素...};` 3. Java 10+ 可使用`var`简化(仍需配合new): ```java var names = new String[]{"板","桃花心","杉木"}; ```
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

foruok

你可以选择打赏

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值