改变范式:从过早的重构和虚假的“可重用性”到适应性、可扩展性和可靠性

改变范式:从过早的重构和虚假的“可重用性”到适应性、可扩展性和可靠性

在软件世界中,人们普遍痴迷于过早的重构以及对虚假可重用性的追求。开发人员(尤其是刚起步的开发人员)经常被教导“可重用性”是圣杯。但不惜一切代价追求可重用性往往会导致过度设计的解决方案,这些解决方案过于通用、过于僵化,并且与当前项目的具体需求相距甚远。事实上,它可能会导致我们通常所说的“抽象地狱”——除非您完全理解系统的每个部分如何以及为什么被抽象以适应通用接口,否则什么都不会真正起作用。

我们建议进行范式转变:不要沉迷于可重用性,让我们关注适应性、可扩展性和可重写性。

在这种情况下,我们不再尝试预测代码库的未来需求(就像算命先生预测未来一样),而是专注于为今天创建一个坚实、灵活的基础,该基础仍有增长空间和随着未来的展开而发展。

过早的重构困境:虚假的可重用性

过早重构的问题在于,它来自于这样的信念:您编写的所有内容都应该可重用。这似乎是一个崇高的目标。然而,可重用性常常会导致不必要的复杂性和不必要的抽象。以创建适用于所有模型的通用 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 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;    }}

现在,不再为每种新型适配器创建一个全新的 baseadapter,我们创建了一个可以轻松扩展和适应未来需求的基础。

扩展新 api 端点的示例:

class OrderAdapter extends APIAdapter {    static baseURL = 'https://api.example.com';    static endpoint = 'orders';}class UserAdapter extends APIAdapter {    static baseURL = 'https://api.example.com';    static endpoint = 'users';}

在这种情况下,如果您需要为一个 api 端点添加特定行为(例如订单的自定义错误处理),您可以覆盖扩展 apiadapter 以适应您的需求无需重构整个系统即可满足需求。

结论:对未来的自己的承诺

在这个新范式中,我们并不试图预测未来的每一个需求或问题。相反,我们专注于建立一个强大、灵活的基础,随着需求的变化和新挑战的出现适应。我们不会根据假设的问题过早地抽象或过度设计解决方案。相反,我们创建工具,这些工具可以随着新需求的出现而不断发展并轻松适应。

关键不是像算命先生一样面向未来,而是创建一个能够可靠地经受时间考验的基础,即使世界发生变化。这是您可以对未来的自己做出的承诺:代码是可靠的、适应性强的,并且可以随着新需求的出现而进行扩展

以上就是改变范式:从过早的重构和虚假的“可重用性”到适应性、可扩展性和可靠性的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1499269.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月19日 21:39:46
下一篇 2025年12月19日 21:39:54

相关推荐

  • 打印队列

    代码来临 2024 年第 5 天 第 1 部分 会有秩序! 这将会是一件很酷的事情。 我喜欢添加的警告,即不应考虑未包含在更新中的页面规则。 我对如何解决这个难题有一个模糊的想法。 但是我需要在这里制定我的策略以保持清晰并确保我准备好编写实际代码。 我希望跌跌撞撞地制定策略 这很有趣。我觉得我知道如…

    好文分享 2025年12月19日
    000
  • 采用声明式数据访问来尊重您作为开发人员的智慧

    在软件开发的世界中,我们经常发现自己在两种范式之间左右为难:命令式和声明式。对于许多开发人员来说,命令式代码的吸引力在于它的简单性——只需逐步编写指令,您就可以确切地知道计算机在做什么。然而,随着复杂性的增加,这种循序渐进的方法变成了分散在代码库中的混乱的逻辑。相比之下,声明式方法旨在让您描述您想要…

    2025年12月19日
    000
  • MeetRoomly 是一款使用 Nextjs Prisma、Tailwind CSS 和 Clerk 轻松管理和预订会议室的应用程序

    见面会 使用 next.js 15、tailwind css、prisma 和 clerk 构建的 meetroomly 应用程序。功能包括用户注册、登录、添加房间、查看房间评论和预订。在开发过程中接受贡献。 入门 克隆存储库:git clone https://github.com/saidmou…

    2025年12月19日
    000
  • 4年前端开发必备技术

    前端开发市场发展迅速,带来了新的工具和实践,改变了创建 Web 应用程序的体验。对于开发人员来说,无论是初学者、全职人员,还是想要了解自己要寻找什么的招聘人员,了解当今不可或缺的技术至关重要。让我们探讨一下 2024 年市场真正发生变化的因素。 1. 现代 JavaScript:坚实的基础 无论你使…

    2025年12月19日
    000
  • React 初学者指南:了解组件

    介绍 嘿,这里是一位开发人员。因为 react 看起来很难而犹豫是否要开始?别担心,我已经帮你解决了。 react 是一个 javascript 库,它使构建用户界面 (ui) 变得更加容易,特别是当您的网站有大量动态内容时。它很强大,但就像任何新技术一样,一开始它可能会令人生畏。事情是这样的:一旦…

    2025年12月19日
    000
  • 在生产中避免控制台日志:稳健日志记录的最佳实践

    简介 日志记录对于调试和监控应用程序至关重要,但不正确的日志记录可能会导致性能问题、安全漏洞和混乱的输出。在本文中,我们将探讨为什么在生产中应避免使用 console.log,并使用示例提供最佳实践。 为什么在生产中应该避免使用 console.log? 性能开销-> 这在我的系统中花费了大约…

    2025年12月19日
    000
  • 使用 Vitest 进行单元测试:下一代测试框架

    为什么选择维泰斯特? vitest 的设计考虑了现代开发。这就是它脱颖而出的原因: 速度 vitest 以 vite 作为基础,利用其闪电般快速的热模块替换 (hmr) 和 esbuild 进行捆绑和转译。结果是: 智能即时监视模式:仅针对受影响的文件重新运行测试,从而实现即时反馈循环。开箱即用的 …

    2025年12月19日
    000
  • 为什么我的 React useEffect 钩子即使在依赖项数组为空的情况下也会运行多次?

    问题在处理 react 项目时,我遇到了 useeffect 挂钩的问题。我的目标是在组件安装时仅从 api 获取数据一次。然而,即使我提供了一个空的依赖项数组,useeffect 仍然运行多次。 这是代码片段: import react, { useeffect, usestate } from …

    2025年12月19日
    000
  • 如何在nodejs环境中使用代理

    配置代理有一个既定的标准。它通过以下环境变量运行: https_proxy:https 流量的代理http_proxy:http 流量的代理no_proxy:不应通过代理运行的 url。 nodejs 的本机获取客户端不提供任何开箱即用的功能,但是您可以使用来自 undici http 客户端的代理…

    2025年12月19日
    000
  • 健壮代码的基本 JavaScript 测试技术

    javascript 测试是软件开发的一个重要方面,可确保代码的可靠性和健壮性。作为一名开发人员,我发现实施全面的测试策略不仅可以尽早发现错误,还可以提高应用程序的整体质量。让我们探索五种基本的 javascript 测试技术,这些技术在我的经验中被证明是非常宝贵的。 单元测试构成了任何可靠测试策略…

    2025年12月19日
    000
  • AI 星座:Nextjs 和 Gemini 的体验

    大家好! 我担任后端开发人员已有几年了,我一直在寻找新技术。最近,我对人工智能和占星学的结合产生了兴趣,并决定使用 Gemini API 创建一个星座预测生成器。 项目: 在这个项目中,我使用 Next.js 作为前端,这是学习新框架的绝佳机会。在后端,我使用 Next.js 和 GEMINI AP…

    2025年12月19日
    000
  • 关心的动画 html css

    ` hj @property –angle { 语法:“”; 初始值:0deg; 继承: false;} @property –x { 语法:“”; 初始值:35%; 继承: false;} @property –shine { 语法:“”; 初始值:透明; 继承…

    2025年12月19日 好文分享
    000
  • js如何自定义事件

    JavaScript 中,自定義事件允許開發人員創建並觸發特定應用事件,讓程式碼做出反應:使用 new CustomEvent() 構造函數創建自定義事件。使用 dispatchEvent() 方法觸發自定義事件。使用 addEventListener() 方法監聽自定義事件,獲取事件名稱、事件監聽…

    2025年12月19日
    000
  • js如何加密

    JavaScript 提供多种加密方法:1. 字符串加密(AES);2. 哈希算法(SHA256);3. 对称加密(TripleDES);4. 非对称加密(rsa);5. 其他方法(base64、Web Crypto API)。 如何使用 JavaScript 加密 JavaScript 是一种广泛…

    2025年12月19日
    000
  • 如何封装js

    封装 JavaScript 代码可提高代码可维护性、可扩展性和可测试性。步骤如下:创建模块。定义私有变量和函数。导出公共接口。导入模块。 如何封装 JavaScript 代码 封装是将代码组织成可重用的模块的过程。这有助于提高代码的可维护性、可扩展性和可测试性。以下是封装 JavaScript 代码…

    2025年12月19日
    000
  • js如何异步

    答案:是的,JavaScript 中存在异步编程,它是一种处理长时间任务的方法,无需等待其完成即可继续执行其他任务。详细描述:异步编程原理:异步编程允许任务在后台运行,而主程序继续执行。JavaScript 中的异步编程:可以使用回调函数、Promise 或 async/await 语法糖实现。优点…

    2025年12月19日
    000
  • Logging System with Proxy and Fetch

    代理对象:fetchlogger 包装了 fetch 函数。它使用 apply trap 来拦截对 fetch 的调用。 请求日志记录:记录请求的 url 和选项。响应处理:记录响应状态、状态文本和 url。克隆响应以确保正文可以被多次读取。 错误处理:捕获并记录提取过程中遇到的任何错误。 使用代理…

    2025年12月19日
    000
  • 使用 MERN 堆栈构建可扩展 Web 应用程序的最佳实践

    介绍: 作为 Web 开发人员,我们的主要目标之一是创建不仅实用而且可扩展的应用程序。可扩展性确保您的应用程序可以随着用户群的增长而增长,处理增加的流量,并随着时间的推移保持性能。在本文中,我将引导您了解使用 MERN 堆栈构建可扩展 Web 应用程序的最佳实践:MongoDB、Express、Re…

    2025年12月19日
    000
  • 如何将交互式图表和图形添加到 Tailwind CSS 管理模板

    管理仪表板模板对于有效管理和可视化数据至关重要。 tailwind css 以其实用性优先的方法而闻名,它简化了设计令人惊叹的管理仪表板的过程。向这些仪表板添加交互式图表和图形可以将原始数据转换为富有洞察力的可视化效果,从而增强整体用户体验。本博客将指导您完成将交互式图表集成到基于 tailwind…

    2025年12月19日 好文分享
    000
  • typescript断言类型

    TypeScript 中的断言类型明确了表达式或变量的类型,帮助编译器理解特定的类型。使用断言类型有两种方法:非空断言操作符 (!) 用于确保变量不会为 null 或 undefined,类型断言语法 () 将所需类型包围在表达式周围。断言类型应谨慎使用,以避免类型错误。在变量确定不为 null 或…

    2025年12月19日
    000

发表回复

登录后才能评论
关注微信