在软件世界中,人们普遍痴迷于过早的重构以及对虚假可重用性的追求。开发人员(尤其是刚起步的开发人员)经常被教导“可重用性”是圣杯。但不惜一切代价追求可重用性往往会导致过度设计的解决方案,这些解决方案过于通用、过于僵化,并且与当前项目的具体需求相距甚远。事实上,它可能会导致我们通常所说的“抽象地狱”——除非您完全理解系统的每个部分如何以及为什么被抽象以适应通用接口,否则什么都不会真正起作用。
我们建议进行范式转变:不要沉迷于可重用性,让我们关注适应性、可扩展性和可重写性。
在这种情况下,我们不再试图预测代码库的未来需求(就像算命先生预测未来一样),而是专注于为今天创建一个坚实、灵活的基础,该基础仍有增长空间和随着未来的展开而发展。
过早的重构困境:虚假的可重用性
过早重构的问题在于,它来自于这样的信念:您编写的所有内容都应该可重用。这似乎是一个崇高的目标。然而,可重用性常常会导致不必要的复杂性和不必要的抽象。以创建适用于所有模型的通用 API 适配器的概念为例。理想情况是该适配器可以处理任何 API 端点、任何数据格式和任何网络条件。但实际上,这意味着您正在为不确定的未来构建框架,而不是有效解决今天的问题。
示例:
让我们看看之前的 BaseAdapter 和 APIAdapter 类:
export class BaseAdapter { constructor(modelClass) { this.modelClass = modelClass; } async get(id) { throw new Error("Method 'get' must be implemented."); } async *all() { throw new Error("Method 'all' must be implemented."); } async query(params = {}) { throw new Error("Method 'query' must be implemented."); } async create(payload) { throw new Error("Method 'create' must be implemented."); } async update(payload) { throw new Error("Method 'update' must be implemented."); } async delete(id) { throw new Error("Method 'delete' must be implemented."); } }
在上面的代码中,BaseAdapter 定义了所有可能的方法,让我们在特定的子类(如 APIAdapter、LocalStorageAdapter 等)中实现它们。这是用于各种适配器的模板。理论上听起来不错,对吧?有一天,如果我们需要连接到新服务或与新存储解决方案集成,我们只需创建另一个子类即可。
但是让我们面对现实:它真的可以重用吗?或者它会变得非常复杂,使您的系统更难以维护、理解和扩展?您真的在构建可以在真实世界中重复使用的东西,还是只是在猜测未来?
转变:从可重用性到适应性、可扩展性和可重写性
不要追求过早的可重用性,我们建议重点关注适应性和可扩展性。这是什么意思?
- 适应性:创建一个可以轻松更改或扩展的基础,而无需重写大部分代码。
- 可扩展性:为新功能留出空间,而无需重构整个架构。
- 可重写性:允许其他人(或将来的您自己)轻松扩展或重写您的代码,而不用冒破坏所有内容的风险。
这并不是要创建适用于当今每种边缘情况的完美可重用代码。相反,我们专注于构建坚实的基础,您可以随着时间的推移在其上进行构建、添加和修改。关键是灵活性,而不是过早的优化。
旧的“界面”范式:预测未来
在 Java(以及许多其他静态类型语言)的旧时代,重点通常是创建接口并使代码“面向未来”。这个想法是提前预测每个场景并围绕它进行设计。
但是,这种方法通常会导致过度设计:为可能永远不会发生的事情进行设计,或者围绕尚未出现的问题构建抽象框架。您实际上正在编写应该是“通用”的代码,而不了解您正在使用的系统的具体需求。
在 Java 中,接口用于定义契约。但是,如果我们将这种思维从“定义契约”转变为简单地设定当前的期望呢?对于当前上下文而言,这是一个清晰且可靠的承诺,无需假设未来会发生什么。
一种新的承诺:对未来的自己的承诺
在我们的新方法中,我们不会像一些神秘的算命师那样对应用程序的未来做出承诺。相反,我们为今天制定明确、可靠的承诺,并确保这些承诺可以在需要时轻松扩展和调整。
这样想:我们不是在预测 5 年后世界会是什么样子;而是在预测 5 年后世界会是什么样子。我们确保我们今天编写的代码能够随着世界的变化而发展和适应。这就像为建筑物打下坚实的地基,确保它足够坚固,能够承受任何变化。
我们做出的“承诺”是对适应性和可扩展性的承诺。我们的目标不是预测未来,而是创建工具,让未来的开发人员(或未来的你)能够根据需要轻松添加、修改或扩展功能。
实际示例:扩展和覆盖适配器
让我们回顾一下我们的 BaseAdapter 和 APIAdapter 示例。我们不会创建尝试处理所有情况的超级通用方法,而是专注于使代码适应性强和易于扩展。
这是 APIAdapter 的快速重新架构:
export class BaseAdapter { constructor(modelClass) { this.modelClass = modelClass; } async get(id) { throw new Error("Method 'get' must be implemented."); } async *all() { throw new Error("Method 'all' must be implemented."); } async query(params = {}) { throw new Error("Method 'query' must be implemented."); } async create(payload) { throw new Error("Method 'create' must be implemented."); } async update(payload) { throw new Error("Method 'update' must be implemented."); } async delete(id) { throw new Error("Method 'delete' must be implemented."); } }
现在,不再为每种新型适配器创建一个全新的 BaseAdapter,我们创建了一个可以轻松扩展和适应未来需求的基础。
扩展新 API 端点的示例:
export class APIAdapter extends BaseAdapter { static baseURL; static headers; static endpoint; async *all(params = {}) { // Custom logic, but easily extensible if needed const url = `${this.baseURL}/${this.endpoint}`; const response = await API.get(url, { params, headers: this.headers }); return response.data; } async query(params = {}) { // Simplified for illustration const url = `${this.baseURL}/${this.endpoint}/search`; const response = await API.get(url, { params }); return response.data; } // Easily extendable for specific cases async customRequest(method, endpoint, params = {}) { const url = `${this.baseURL}/${endpoint}`; const response = await API[method](url, { params }); return response.data; } }
在这种情况下,如果您需要为一个 API 端点添加特定行为(例如订单的自定义错误处理),您可以覆盖或扩展 APIAdapter 以适应您的需求无需重构整个系统即可满足需求。
结论:对未来的自己的承诺
在这个新范式中,我们并不试图预测未来的每一个需求或问题。相反,我们专注于建立一个强大、灵活的基础,随着需求的变化和新挑战的出现适应。我们不会根据假设的问题过早地抽象或过度设计解决方案。相反,我们创建工具,这些工具可以随着新需求的出现而不断发展并轻松适应。
关键不是像算命先生一样面向未来,而是创建一个能够可靠地经受时间考验的基础,即使世界发生变化。这是您可以对未来的自己做出的承诺:代码是可靠的、适应性强的,并且可以随着新需求的出现而进行扩展。
以上是改变范式:从过早的重构和虚假的'可重用性”到适应性、可扩展性和可靠性的详细内容。更多信息请关注PHP中文网其他相关文章!

Python和JavaScript的主要区别在于类型系统和应用场景。1.Python使用动态类型,适合科学计算和数据分析。2.JavaScript采用弱类型,广泛用于前端和全栈开发。两者在异步编程和性能优化上各有优势,选择时应根据项目需求决定。

选择Python还是JavaScript取决于项目类型:1)数据科学和自动化任务选择Python;2)前端和全栈开发选择JavaScript。Python因其在数据处理和自动化方面的强大库而备受青睐,而JavaScript则因其在网页交互和全栈开发中的优势而不可或缺。

Python和JavaScript各有优势,选择取决于项目需求和个人偏好。1.Python易学,语法简洁,适用于数据科学和后端开发,但执行速度较慢。2.JavaScript在前端开发中无处不在,异步编程能力强,Node.js使其适用于全栈开发,但语法可能复杂且易出错。

javascriptisnotbuiltoncorc; saninterpretedlanguagethatrunsonenginesoftenwritteninc.1)javascriptwasdesignedAsalightweight,解释edganguageforwebbrowsers.2)Enginesevolvedfromsimpleterterterpretpreterterterpretertestojitcompilerers,典型地提示。

JavaScript可用于前端和后端开发。前端通过DOM操作增强用户体验,后端通过Node.js处理服务器任务。1.前端示例:改变网页文本内容。2.后端示例:创建Node.js服务器。

选择Python还是JavaScript应基于职业发展、学习曲线和生态系统:1)职业发展:Python适合数据科学和后端开发,JavaScript适合前端和全栈开发。2)学习曲线:Python语法简洁,适合初学者;JavaScript语法灵活。3)生态系统:Python有丰富的科学计算库,JavaScript有强大的前端框架。

JavaScript框架的强大之处在于简化开发、提升用户体验和应用性能。选择框架时应考虑:1.项目规模和复杂度,2.团队经验,3.生态系统和社区支持。

引言我知道你可能会觉得奇怪,JavaScript、C 和浏览器之间到底有什么关系?它们之间看似毫无关联,但实际上,它们在现代网络开发中扮演着非常重要的角色。今天我们就来深入探讨一下这三者之间的紧密联系。通过这篇文章,你将了解到JavaScript如何在浏览器中运行,C 在浏览器引擎中的作用,以及它们如何共同推动网页的渲染和交互。JavaScript与浏览器的关系我们都知道,JavaScript是前端开发的核心语言,它直接在浏览器中运行,让网页变得生动有趣。你是否曾经想过,为什么JavaScr


热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

Video Face Swap
使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热门文章

热工具

SecLists
SecLists是最终安全测试人员的伙伴。它是一个包含各种类型列表的集合,这些列表在安全评估过程中经常使用,都在一个地方。SecLists通过方便地提供安全测试人员可能需要的所有列表,帮助提高安全测试的效率和生产力。列表类型包括用户名、密码、URL、模糊测试有效载荷、敏感数据模式、Web shell等等。测试人员只需将此存储库拉到新的测试机上,他就可以访问到所需的每种类型的列表。

PhpStorm Mac 版本
最新(2018.2.1 )专业的PHP集成开发工具

SublimeText3汉化版
中文版,非常好用

mPDF
mPDF是一个PHP库,可以从UTF-8编码的HTML生成PDF文件。原作者Ian Back编写mPDF以从他的网站上“即时”输出PDF文件,并处理不同的语言。与原始脚本如HTML2FPDF相比,它的速度较慢,并且在使用Unicode字体时生成的文件较大,但支持CSS样式等,并进行了大量增强。支持几乎所有语言,包括RTL(阿拉伯语和希伯来语)和CJK(中日韩)。支持嵌套的块级元素(如P、DIV),

EditPlus 中文破解版
体积小,语法高亮,不支持代码提示功能