编码 10000 个小时后,开发者悟了:“不要急于发布!”
liuian 2025-09-04 11:56 22 浏览
【CSDN 编者按】在软件开发的道路上,时间是最好的老师。
根据“一万小时定律”,要成为某个领域的专家,通常需要大约一万小时的刻意练习。
本文作者身为一名程序员,也经历了一万小时的编程,最终悟出了一个道理:慢即是快,重视架构设计和代码质量,确保每一行代码都经得起时间的考验。
作者 | Sotiris Kourouklis 翻译 | 郑丽媛
出品 | CSDN(ID:CSDNnews)
我已经做了 7 年多的程序员了,在这期间,我在后端、前端和 DevOps 方面参与了无数个项目。尽管如此,我并不认为自己是一名伟大的程序员;有很多人不仅比我聪明,经验也更丰富。不过这些年来我也总结出了一些经验,让我在编程道路上不断进步,并构建出更可靠且易于维护的软件。
“学会放慢速度,反而能让我编码更快、发布更多,并在整体上更加高效。”
这不仅仅是多年的编程经验,也是我从生活中学到的教训:你必须时刻保持缓慢,不要急躁。
软件开发中的头号问题
很多人在开始编程时,总认为伟大的程序员就像会魔法,他们能以一种无人能懂的独特方式去构建应用程序——但这与事实完全不符。如果仔细查看那些伟大程序员的代码,你会发现它们其实非常简单、很好理解。
不是说在开发应用程序时使用快速、炫酷或使用最尖端的技术,你才能被认为是一名优秀的程序员。管理者也常常犯这个错误,他们老是根据一些不切实际的标准来进行招聘。在我看来,要求应聘者从零开始构建一个应用程序的编程测试是一种相当糟糕的评估方法,它无法全面反映一个人的能力。相比之下,白板面试其实更好,因为至少它可以考察你的智商和思维方式。
那么接下来,让我们谈谈每个人都应该关注但常常忽视的头号问题:
开发者体验
在绝大多数项目中,开发者体验是最重要的事情。每个人都希望快速发布以实现盈利,但最终他们往往在一个月内完成了 90% 的应用程序开发,而剩下的 10% 却需要三个月才能完成。
我理解,开发者总是对新项目充满热情,想要尽快展示成果以获得满足感,并让经理满意。虽然短期内这确实能让经理高兴,但从长远来看,每个人都会陷入恐慌,并在 4-5 年后考虑重构甚至从头开始重建那款应用程序——这就是为什么实际创建功能变得非常困难,而将其推向生产更是难上加难,于是便产生了滚雪球效应。
下面让我们来看两个代码示例:两个控制器,用于从数据库中获取热门用户并为其附加表情符号(来自我的一个开源项目 reporanger.xyz 的代码)。
这是从路由调用的控制器,其中包含所有功能以及一个 try-catch 块,用于检查是否存在错误:
// users.controller.tsconst getTrendingUsers = async (_req: Request, res: Response, _next: NextFunction) => {try {const events = await GithubEvent.find({where: { event_date: MoreThan(new Date(Date.now - 24 * 60 * 60 * 1000)) },order: { event_size: 'DESC' },take: 3,});const users = await Username.find({ where: { id: In(events.map((event) => event.username_id)) } });const topUsers = await getTopUsers(3);const trendingUsers = await Promise.all(users.map(async (user) => ({...user,emoji: await emojiService.getEmoji(user.score, topUsers),})),);res.status(200).json({status: 200,message: 'Trending users fetched successfully',data: trendingUsers,error: '',success: true,});} catch (error) {res.status(500).json({status: 500,message: 'Error fetching trending users',data: ,error: error.message,success: false,});}};
下面是同样的控制器,但我们应用了一个简单原则:单一职责原则。
我们将代码拆分成了 4 个更小且可复用的文件:
● user.controller.ts
● user.service.ts
● async.util.ts
● response.util.ts
经过这样的优化后,带来的好处远超我们的想象!
我们可以在应用程序的任何地方重用这些代码。比如,在定时任务中我们可以调用
userService.getTrendingUsers 来获取热门用户信息。
我们移除了所有 try-catch 语句块,让代码更加简洁。同时,对于每个错误我们都进行了日志记录(例如使用 logger('error', error))。这样一来就可以很容易地创建一个错误服务,将所有错误信息存储到数据库中,为未来的应用场景做好准备。
另外,通过统一所有控制器的响应方式(如使用 resFn)也非常重要,它可以确保所有请求都能以相同的格式返回响应,正如下面示例中的简单泛型所示。
如此一来,开发一个新的控制器就变得轻松十倍,因为我们整个应用程序的编码架构非常一致。即使后来由其他开发者编写代码,也不会影响整体的编码风格。
// user.controller.tsconst getTrendingUsers = asyncFn(async (_req: Request, res: Response, _next: NextFunction) => {const trendingUsers = await usernameService.getTrendingUsers;resFn(res, {status: 200,message: 'Trending users fetched successfully',data: trendingUsers,error: '',success: true,});});// user.service.tsconst getTrendingUsers = async => {const events = await GithubEvent.find({where: { event_date: MoreThan(new Date(Date.now - 24 * 60 * 60 * 1000)) },order: { event_size: 'DESC' },take: 3,});const users = await Username.find({ where: { id: In(events.map((event) => event.username_id)) } });const topUsers = await getTopUsers(3);const usersWithEmoji = await Promise.all(users.map(async (user) => ({...user,emoji: await emojiService.getEmoji(user.score, topUsers),})),);return usersWithEmoji;};// async.util.tsexport const asyncFn = (fn: asyncPropsFunction) => async (req: Request, res: Response, next: NextFunction) => {try {await fn(req, res, next);} catch (error) {logger('error', error);next(error);}};// response.util.tsexport const resFn = (res: Response, { status, error, data, message, success }: IResponse<any>) => {const suc = success !== undefined ? success : true;res.status(status).json({error,data,message,success: suc,status,});};// response.interface.tsexport interface IResponse<T> {status: number;message: string;data: T | any;error: string;success: boolean;}
设想一下:假如我们从一开始就没有实现良好的架构,那么在未来想要对应用程序做一些小的改动时会怎样呢?哪怕只是想记录错误,我们也需要去每一个控制器中添加 logger('error', error);或者我们还想在响应中增加一个额外字段,比如 metadata?那必将会是一场噩梦。
首先进行重构
我觉得,重构应当在编写代码之前就做好:因为每个应用程序最终都需要重构。例如,对于一个拥有超过 70,000 行代码的大型软件项目来说,重构可能需要 30-40 个小时,在此过程中还会引入大量错误和 bug。
最终,你可能会不小心破坏应用程序,或者再花费 20 个小时进行测试。而且,当你的项目达到如此大的规模时,重构效果也不会像一开始就做重构那么理想。
我的建议是,在最初投入 40-50 个小时来规划和重构。只需创建几个控制器,集思广益地考虑如何在未来扩展这些控制器,然后进行重构,之后再继续开发。我知道,一开始你的经理可能会抱怨,因为你花了 50 个小时编码,但几乎没有什么实质性成果可以展示,只有一个能够良好扩展但没人能立即理解的架构。但如果条件允许的话,我觉得还是应该这样做。这不仅会为你自己,也会为未来的开发者省去很多麻烦。
单元测试同样非常重要。不要过分追求覆盖率,保持 60% 以上的覆盖率就足够了,这将在未来帮你避免大量潜在的 bug。
提交前检查
这一点非常重要。对于 JavaScript/TypeScript 项目,我们可以用 Husky。当然,针对不同的语言或框架还有很多其他选择。简单来说,Husky 是一个工具,它会在你提交代码之前运行一些命令。如果出现错误,那提交就不会通过。下面是一个来自 reporanger.xyz 的配置示例:
1、Lint Check(代码风格检查)
2、Run Tests(运行测试)
3、Prettify(代码格式化)
这三个简单的步骤,就能让你的代码库质量提升 10 倍。
#!/bin/shnpx eslint --max-warnings=0 src api/src || {echo "ESLint check failed. Commit aborted."exit 1}cd ./api && npx jest || {echo "Tests failed. Commit aborted."exit 1}cd .. && npx prettier --write .git update-index --again
简而言之
有很多方法可以用来改进你的代码,而我上面提到的这些做法其实并不难实现。尤其现在有了大模型(LLM)的帮助,很多问题往往不是你能不能做,而是你是否愿意去做。
抛开无聊的情绪,你会发现这样做会带来很多好处。开始带着热爱去编码,而不仅仅是为了钱。如果你能做到这一点,你会赚到更多的钱,让你的同事开心,你的经理也会感谢你,因为——编程不仅仅是写代码,更是一种架构设计。
相关推荐
- 搭建一个20人的办公网络(适用于20多人的小型办公网络环境)
-
楼主有5台机上网,则需要一个8口路由器,组网方法如下:设备:1、8口路由器一台,其中8口为LAN(局域网)端口,一个WAN(广域网)端口,价格100--400元2、网线N米,这个你自己会看了:)...
- 笔记本电脑各种参数介绍(笔记本电脑各项参数新手普及知识)
-
1、CPU:这个主要取决于频率和二级缓存,频率越高、二级缓存越大,速度越快,现在的CPU有三级缓存、四级缓存等,都影响相应速度。2、内存:内存的存取速度取决于接口、颗粒数量多少与储存大小,一般来说,内...
- 汉字上面带拼音输入法下载(字上面带拼音的输入法是哪个)
-
使用手机上的拼音输入法打成汉字的方法如下:1.打开手机上的拼音输入法,在输入框中输入汉字的拼音,例如“nihao”。2.根据输入法提示的候选词,选择正确的汉字。例如,如果输入“nihao”,输...
- xpsp3安装版系统下载(windowsxpsp3安装教程)
-
xpsp3纯净版在采用微软封装部署技术的基础上,结合作者的实际工作经验,融合了许多实用的功能。它通过一键分区、一键装系统、自动装驱动、一键设定分辨率,一键填IP,一键Ghost备份(恢复)等一系列...
- 没有备份的手机数据怎么恢复
-
手机没有备份恢复数据方法如下1、使用数据线将手机与电脑连接好,在“我的电脑”中可以看到手机的盘符。 2、将手机开启USB调试模式。在手机设置中找到开发者选项,然后点击“开启USB调试模式”。 3、...
- 电脑怎么激活windows11专业版
-
win11专业版激活方法有多种,以下提供两种常用的激活方式:方法一:使用激活密钥激活。在win11桌面上右键点击“此电脑”,选择“属性”选项。进入属性页面后,点击“更改产品密钥或升级windows”。...
- 华为手机助手下载官网(华为手机助手app下载专区)
-
华为手机助手策略调整,已不支持从应用市场下载手机助手,目前华为手机助手是需要在电脑上下载或更新手机助手到最新版本,https://consumer.huawei.com/cn/support/his...
- 光纤线断了怎么接(宽带光纤线断了怎么接)
-
宽带光纤线断了可以重接,具体操作方法如下:1、光纤连接的时候要根据束管内,同色相连,同芯相连,按顺序进行连接,由大到小。一般有三种连接方法,分别是熔接、活动连接和机械连接。2、连接的时候要开剥光缆,抛...
- win7旗舰版和专业版区别(win7旗舰版跟专业版)
-
1、功能区别:Win7旗舰版比专业版多了三个功能,分别是Bitlocker、BitlockerToGo和多语言界面; 2、用途区别:旗舰版的功能是所有版本中最全最强大的,占用的系统资源,...
- 万能连接钥匙(万能wifi连接钥匙下载)
-
1、首先打开wifi万能钥匙软件,若手机没有开启WLAN,就根据软件提示打开WLAN开关;2、打开WLAN开关后,会显示附近的WiFi,如果知道密码,可点击相应WiFi后点击‘输入密码’连接;3、若不...
- 雨林木风音乐叫什么(雨林木风是啥)
-
雨林木风的创始人是陈年鑫先生。陈年鑫先生于1999年创立了雨林木风公司,其初衷是为满足中国市场对高品质、高性能电脑的需求。在陈年鑫先生的领导下,雨林木风以技术创新、产品质量和客户服务为核心价值,不断推...
- aics6序列号永久序列号(aics6破解序列号)
-
关于AICS6这个版本,虽然是比较久远的版本,但是在功能上也是十分全面和强大的,作为一名平面设计师的话,AICS6的现有的功能已经能够应付几乎所有的设计工作了……到底AICC2019的功能是不是...
- 手机可以装电脑系统吗(手机可以装电脑系统吗怎么装)
-
答题公式1:手机可以通过数据线或无线连接的方式给电脑装系统。手机安装系统需要一定的技巧和软件支持,一般需要通过数据线或无线连接的方式与电脑连接,并下载相应的软件和系统文件进行安装。对于大部分手机用户来...
- 一周热门
- 最近发表
- 标签列表
-
- python判断字典是否为空 (50)
- crontab每周一执行 (48)
- aes和des区别 (43)
- bash脚本和shell脚本的区别 (35)
- canvas库 (33)
- dataframe筛选满足条件的行 (35)
- gitlab日志 (33)
- lua xpcall (36)
- blob转json (33)
- python判断是否在列表中 (34)
- python html转pdf (36)
- 安装指定版本npm (37)
- idea搜索jar包内容 (33)
- css鼠标悬停出现隐藏的文字 (34)
- linux nacos启动命令 (33)
- gitlab 日志 (36)
- adb pull (37)
- python判断元素在不在列表里 (34)
- python 字典删除元素 (34)
- vscode切换git分支 (35)
- python bytes转16进制 (35)
- grep前后几行 (34)
- hashmap转list (35)
- c++ 字符串查找 (35)
- mysql刷新权限 (34)
