编码 10000 个小时后,开发者悟了:“不要急于发布!”
liuian 2025-09-04 11:56 25 浏览
【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)的帮助,很多问题往往不是你能不能做,而是你是否愿意去做。
抛开无聊的情绪,你会发现这样做会带来很多好处。开始带着热爱去编码,而不仅仅是为了钱。如果你能做到这一点,你会赚到更多的钱,让你的同事开心,你的经理也会感谢你,因为——编程不仅仅是写代码,更是一种架构设计。
相关推荐
- office和visio安装顺序(office和visio怎么一起安装)
-
在某些情况下,安装Visio可能会发生与Office365冲突的问题。这是因为Visio和Office365具有不同的版本,可能会导致安装时出现错误或兼容性问题。为了避免这种冲突,...
- 电脑中病毒的原因(电脑中病毒正常吗)
-
电脑中毒的原因有以下几方面:1.网页被挂病毒。2.电脑裸奔,无防病毒软件。3.执行一些不安全的程序。4.U盘等不安全介质。5.电脑漏洞不及时补,被后台种毒。为了电脑不中病毒要注意以下几方面:1.更新系...
- 手机psd转换成jpg最简单方式
-
可以使用photoshop工具,方法如下:1、首先打开PS软件,然后选择自己需要的JPG格式的图片,在PS中打开。2、接下来先按快捷键“Ctrl+j”将图片复制出来,防止后面操作对原图片有损...
- win7提示激活码过期怎么办(win7激活已过期)
-
以win7为例,出现这样的问题原因分析:电脑的win7系统激活过又重新提示要激活的原因是因为微软对网络上的秘钥进行封杀所以导致我们激活无效。具体的解决方法:1、我们打开dos命令窗口,在创立中输入“s...
- 联想笔记本光驱驱动下载(联想电脑光驱驱动器在哪)
-
开机时进入BIOS,具体按什么牌子不同,按键也不同,开机有提示的,选择启动项,把光驱启动的顺序放到第一.按F10保存,重新启动就是光驱启动啦不需要设置光驱驱动,笔记本自带光驱驱动光驱是电脑的硬件设备,...
- win10装机必备实用软件(win10电脑装机必备软件)
-
1、office大部分的版本如office2007、office2000、office2011、office2013、office2016、office365等都支持win10。2、需要注意...
- 迅雷无法下载的链接用什么下载
-
1.可以使用其他下载工具代替迅雷。2.迅雷可能无法下载的原因有很多,比如网络问题、软件故障等。其他下载工具可以提供类似的功能,但可能具有更好的稳定性和兼容性。3.一些常见的替代迅雷的下载工具包括...
- apple官方网站(apple官方网站旗舰店)
-
1、首先打开浏览器,输入https://www.apple.com/;2、即可浏览苹果官网。 苹果公司(AppleInc.)是美国一家高科技公司。由史蒂夫·乔布斯、斯蒂夫·沃兹尼亚克和罗·韦恩(R...
- 哪些手机用鸿蒙系统(都什么手机能用鸿蒙系统)
-
截至目前,国内有以下几款手机品牌可以装鸿蒙系统:1.华为:华为Mate40系列、P40系列、Mate30系列、MatePadPro系列等。2.荣耀:荣耀V40、荣耀30系列、荣耀X10系列等...
- 手机u盘读不出来了怎么修复(手机u盘读取不出来)
-
1、手机不支持OTG功能,所以将U盘连接到手机后,手机无法识别U盘的内容,因此显示不了;这种情况只能换台支持OTG功能的手机来连接U盘才行。2、手机支持OTG功能,但是使用的OTG线质量有问题导致无法...
- 笔记本散热器买哪种好(笔记本散热器买哪种好贴吧)
-
散热器有十大品牌:九州风神、超频三,酷冷至尊Tt、AVC、思民、捷冷、安钛克Antec、安耐美Enermax、海盗船Corsair。能位列十大品牌,每一种的质量和功能都有保障。、目前网上销量最高的是九...
-
- 打印机驱动一直安装失败(打印机驱动一直安装失败怎么办)
-
打印机驱动程序安装失败需要对电脑进行其他设置,详细步骤如下:1,在电脑桌面上找到【计算机】并用鼠标右击。2,右击后在出现的选项中找到【管理】选项并点击打开。3,接下里会进入到计算机控制台界面,在这里要根据自己的电脑选择64位或者32位,选择...
-
2026-01-14 12:55 liuian
- ctrl加谁是截图(ctrl和什么键可以截图)
-
第一种:Ctrl+PrScrn使用这个组合键截屏,获得的是整个屏幕的图片第二种:Alt+PrScrn这个组合键截屏,获得的结果是当前窗口的图片第三种:打开qq,使用快捷键Ctrl+...
- 一周热门
-
-
飞牛OS入门安装遇到问题,如何解决?
-
如何在 iPhone 和 Android 上恢复已删除的抖音消息
-
Boost高性能并发无锁队列指南:boost::lockfree::queue
-
大模型手册: 保姆级用CherryStudio知识库
-
用什么工具在Win中查看8G大的log文件?
-
如何在 Windows 10 或 11 上通过命令行安装 Node.js 和 NPM
-
威联通NAS安装阿里云盘WebDAV服务并添加到Infuse
-
Trae IDE 如何与 GitHub 无缝对接?
-
idea插件之maven search(工欲善其事,必先利其器)
-
如何修改图片拍摄日期?快速修改图片拍摄日期的6种方法
-
- 最近发表
- 标签列表
-
- 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)
