0

0

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

霞舞

霞舞

发布时间:2024-12-10 08:07:08

|

1144人浏览过

|

来源于dev.to

转载

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

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

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

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


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

过早重构的问题在于,它来自于这样的信念:您编写的所有内容都应该可重用。这似乎是一个崇高的目标。然而,可重用性常常会导致不必要的复杂性和不必要的抽象。以创建适用于所有模型的通用 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 等)中实现它们。这是用于各种适配器的模板。理论上听起来不错,对吧?有一天,如果我们需要连接到新服务或与新存储解决方案集成,我们只需创建另一个子类即可。

但是让我们面对现实:它真的可以重用吗?或者它会变得复杂起来,让你的系统更难维护、理解和扩展吗?您真的在构建可以在真实世界中重复使用的东西,还是只是在猜测未来?


转变:从可重用性到适应性、可扩展性和可重写性

不要追求过早的可重用性,我们建议重点关注适应性可扩展性。这是什么意思?

  1. 适应性:创建一个可以轻松更改或扩展的基础,而无需重写大部分代码。
  2. 可扩展性:为新功能留出空间,而无需重构整个架构。
  3. 可重写性:允许其他人(或将来的您自己)轻松扩展或重写您的代码,而不用冒破坏所有内容的风险。

这并不是要创建适用于当今每种边缘情况的完美可重用代码。相反,我们专注于构建坚实的基础,您可以随着时间的推移在其上进行构建、添加和修改。关键是灵活性,而不是过早的优化。


旧的“界面”范式:预测未来

在 java(以及许多其他静态类型语言)的旧时代,重点通常是创建接口并使代码“面向未来”。这个想法是提前预测每个场景并围绕它进行设计。

但是,这种方法通常会导致过度设计:为可能永远不会发生的事情进行设计,或者围绕尚未出现的问题构建抽象框架。您实际上正在编写应该是“通用”的代码,而不了解您正在使用的系统的具体需求。

知识吐司
知识吐司

专注K12教育的AI知识漫画生成工具

下载

在 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 以适应您的需求无需重构整个系统即可满足需求。


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

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

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

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

866

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

745

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

741

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

420

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

447

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

431

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16947

2023.08.03

c++ 根号
c++ 根号

本专题整合了c++根号相关教程,阅读专题下面的文章了解更多详细内容。

70

2026.01.23

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号