DIP的实际举例

SOLID原则。

依赖倒置原则(DIP)的核心是高层模块不应该依赖于低层模块,二者都应该依赖于抽象(接口或抽象类)

例如,随着业务的发展,订单总金额的计算规则可能需要根据不同的客户类型或促销活动进行调整。如果最初没有使用接口来抽象计算逻辑,那么可能需要直接修改具体的业务代码,这会增加代码的修改风险和维护成本。而使用接口可以方便地实现不同的计算策略,通过依赖注入等方式进行切换,提高系统的灵活性和可扩展性。
接下来我们逐步优化,把他变得满足SRP和OCP、DIP原则了。

循环累加

public class Order {
    private double[] productPrices;

    public Order(double[] productPrices) {
        this.productPrices = productPrices;
    }

    // 计算订单总金额的方法
    public double calculateTotalAmount() {
        double total = 0;
        for (double price : productPrices) {
            total += price;
        }
        return total;
    }
}

当业务发展,需要根据不同客户类型(比如普通客户、VIP 客户)和促销活动(比如满减、折扣)来调整订单总金额的计算规则时,就需要直接修改 calculateTotalAmount 方法的代码。例如,VIP 客户享受 9 折优惠,满 100 减 20 的促销活动,代码修改如下:

public class Order {
    private double[] productPrices;
    private String customerType;

    public Order(double[] productPrices, String customerType) {
        this.productPrices = productPrices;
        this.customerType = customerType;
    }

    // 计算订单总金额的方法,修改后变得复杂
    public double calculateTotalAmount() {
        double total = 0;
        for (double price : productPrices) {
            total += price;
        }
        if ("VIP".equals(customerType)) {
            total *= 0.9;
        }
        if (total >= 100) {
            total -= 20;
        }
        return total;
    }
}

这种直接修改的方式,随着业务规则的不断增加和变化,代码会变得越来越复杂和难以维护,修改一处可能会影响到其他部分的逻辑,增加了出错的风险。

抽象工厂

我们发现,订单类和计算金额的修改原因、修改速率不一样,根据SRP原则,利用抽象工厂模式,把他抽出来,如下。

// 订单类
public class Order {
    private double[] productPrices;
    private String customerType;

    public Order(double[] productPrices, String customerType) {
        this.productPrices = productPrices;
        this.customerType = customerType;
    }

    // 计算总金额,依赖工厂类
    public double calculateTotalAmount() {
        return OrderTotalCalculatorFactory.calculateTotal(productPrices, customerType);
    }
}

// 订单总金额计算工厂类
public class OrderTotalCalculatorFactory {
    public static double calculateTotal(double[] productPrices, String customerType) {
        if ("Regular".equals(customerType)) {
            double total = 0;
            for (double price : productPrices) {
                total += price;
            }
            return total;
        } else if ("VIP".equals(customerType)) {
            double total = 0;
            for (double price : productPrices) {
                total += price;
            }
            total *= 0.9;
            return total;
        } else {
            double total = 0;
            for (double price : productPrices) {
                total += price;
            }
            if (total >= 100) {
                total -= 20;
            }
            return total;
        }
    }
}

引入抽象类

如果再有新的计费类型时,我们需要修改OrderTotalCalculatorFactory ,再添加一个if-else,然后在里面写代码,违反了OCP原则。所以我们不把具体代码写到OrderTotalCalculatorFactory 里,而是再抽一层,可以是接口,也可以不是。(这样不是说没有改动,但是改动点小了。是扩展本类,而不是修改)
但是为了专业(把相同的抽出来,作为上层),我们没有理由不写成接口。
此外如果再想加一个过期时间校验功能(我们的优惠、会员等都是有时间的),先校验,然后再计算金额。这个时候就体现出接口的好处来了。

这样就是所谓的DIP了:由原来的order订单依赖于金额计算,现在加了一个接口,二者都依赖于这个接口了。

// 订单总金额计算策略接口
public interface OrderTotalCalculator {
    double calculateTotal(double[] productPrices);
}

// 普通客户计算策略实现类
public class RegularCustomerCalculator implements OrderTotalCalculator {
    @Override
    public double calculateTotal(double[] productPrices) {
        double total = 0;
        for (double price : productPrices) {
            total += price;
        }
        return total;
    }
}

// VIP客户计算策略实现类
public class VIPCustomerCalculator implements OrderTotalCalculator {
    @Override
    public double calculateTotal(double[] productPrices) {
        double total = 0;
        for (double price : productPrices) {
            total += price;
        }
        total *= 0.9;
        return total;
    }
}

// 工厂类
public class OrderTotalCalculatorFactory {
    public static OrderTotalCalculator createCalculator(String customerType) {
        if ("Regular".equals(customerType)) {
            return new RegularCustomerCalculator();
        } else if ("VIP".equals(customerType)) {
            return new VIPCustomerCalculator();
        } 
        // 目前只有这两种情况,后续可扩展
        return null;
    }
}


// 组合优于继承
public class Order {
    private double[] productPrices;
    private String customerType;
    private OrderTotalCalculator calculator;

    public Order(double[] productPrices, String customerType) {
        this.productPrices = productPrices;
        this.customerType = customerType;
        this.calculator = OrderTotalCalculatorFactory.createCalculator(customerType);
    }

    public double calculateTotalAmount() {
        return calculator.calculateTotal(productPrices);
    }
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/975742.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

如何在 SpringBoot 项目创建并使用 Redis 的详细介绍

本文是博主整理项目时整理出来的,项目使用 SpringBoot 框架,使用 Redis 作为缓存组件,用于缓存部分热点接口数据。 文章目录 一、Redis 的前置配置1、引入依赖2、单机配置3、集群配置4、两种配置路径解读 二、创建 Redis 配置类1、简易版 Red…

快速入门——第三方组件element-ui

学习自哔哩哔哩上的“刘老师教编程”,具体学习的网站为:10.第三方组件element-ui_哔哩哔哩_bilibili,以下是看课后做的笔记,仅供参考。 第一节 组件间的传值 组件可以有内部Data提供数据,也可由父组件通过prop方式传…

使用 Grafana 监控 Spring Boot 应用

随着软件开发领域的不断发展,监控和可观测性已成为确保系统可靠性和性能的关键实践。Grafana 是一个功能强大的开源工具,能够为来自各种来源的监控数据提供丰富的可视化功能。在本篇博客中,我们将探讨如何将 Grafana 与 Spring Boot 应用程序…

《深度学习》——RNN网络简单介绍

文章目录 RNN网络简介工作原理网络结构训练方法应用领域 RNN网络简介 循环神经网络(Recurrent Neural Network,RNN)是一种专门用于处理序列数据的神经网络,在自然语言处理、语音识别、时间序列预测等领域有广泛应用。 RNN 是一种…

深入解析JVM垃圾回收机制

1 引言 本节常见面试题 如何判断对象是否死亡(两种方法)。简单的介绍一下强引用、软引用、弱引用、虚引用(虚引用与软引用和弱引用的区别、使用软引用能带来的好处)。如何判断一个常量是废弃常量如何判断一个类是无用的类垃圾收…

python网络安全怎么学 python做网络安全

🍅 点击文末小卡片 ,免费获取网络安全全套资料,资料在手,涨薪更快 众所周知,python是近几年比较火的语言之一,它具有简单易懂、免费开源、可移植、可扩展、丰富的第三方库函数等特点,Java需要大…

使用excel中的VBA合并多个excel文件

需求是这样的: 在Windows下,用excel文件让多个小组填写了统计信息,现在我需要把收集的多个文件汇总到一个文件中,前三行为标题可以忽略,第四行为收集信息的列名,处理每一行数据的时候,发现某一行…

【算法】2022年第十三届蓝桥杯大赛软件类省赛Java大学C组真题

个人主页:NiKo 算法专栏:算法设计与分析 目录 题目 2680:纸张尺寸 题目 2664:求和 题目 2681: 矩形拼接 题目 2665: 选数异或 题目 2682: GCD 题目 2667: 青蛙过河 题目 2683: 因数平方和 题目 2668: 最长不下降子序列 题目 2680:纸张尺寸 题目…

2025雅森北京展今天开幕,全景展现新能源汽车时代

推动汽车后市场全产业链向新向优转型升级 在2月21日举行的第36届中国国际汽车服务用品及设备展览会,中国国际新能源汽车技术、零部件及服务展览会(雅森北京展CIAACE)上,雅森国际总裁谢宇表示,当前正处在汽车行业新旧动…

PHP.INI的作用以及如何设置

php.ini是网站服务器上的一个重要配置文件,它包含了决定PHP运行环境特性的关键设置,直接关系到网站能否顺畅运行。每次 PHP 启动时,系统都会读取这个文件,加载相应的配置并将其应用到网站的脚本上。 尽管服务器通常预装了一些默认…

新版IDEA配置 Tomcat

配置tomcat 先检查自己的tomcat是否已经安装完成以及是否有其他问题,这里不对Tomcat方面的问题进行讲解。 新版IDEA需要在已有项目中添加Web部分的项目结构。 安装完成之后,打开要创建Web项目的项目(这里演示使用新建项目) 第一步,先给项…

整理好的java面试八大常用算法

原创不易,如果有转载需要的话,请在首行附上本文地址,谢谢。 文中整理了八大常见的排序算法,冒泡排序、选择排序、快速排序、插入排序、堆排序、希尔排序、归并排序和基数排序的简单思想,每种算法配有动图分析和相应的…

Baklib智能平台:数据驱动下的企业知识安全与协作

内容概要 在数字化转型加速渗透的今天,企业知识资产的智能化管理与安全防护已成为核心竞争力构建的关键环节。Baklib作为新一代知识中台(Knowledge Hub)的典型代表,以数据驱动技术为核心引擎,重构了企业知识管理的底层…

深度学习-7.超参数优化

Deep Learning - Lecture 7 Hyperparameter Optimization 简介超参数搜索用于超参数选择的贝叶斯优化启发性示例贝叶斯优化 引用 本节目标: 解释并实现深度学习中使用的不同超参数优化方法,包括: 手动选择网格搜索随机搜索贝叶斯优化 简介 …

【漫画机器学习系列】102.带泄露线性整流函数(Leaky ReLU)

Leaky ReLU(带泄露线性整流函数)详解 1. 什么是 Leaky ReLU? Leaky ReLU(带泄露线性整流函数)是一种改进的 ReLU(Rectified Linear Unit,线性整流单元)激活函数。与标准 ReLU 不同…

GeoHD - 一种用于智慧城市热点探测的Python工具箱

GeoHD - 一种用于智慧城市热点探测的Python工具箱 详细原理请参考:Yan, Y., Quan, W., Wang, H., 2024. A data‐driven adaptive geospatial hotspot detection approach in smart cities. Trans. GIS tgis.13137. 代码下载:下载 1. 简介 在城市数据…

Github 2025-02-23 php开源项目日报 Top9

根据Github Trendings的统计,今日(2025-02-23统计)共有9个项目上榜。根据开发语言中项目的数量,汇总情况如下: 开发语言项目数量PHP项目9JavaScript项目2Shell项目1TypeScript项目1Blade项目1Java项目1ASP项目1Vue项目1Laravel:表达力和优雅的 Web 应用程序框架 创建周期:…

使用大语言模型(Deepseek)构建一个基于 SQL 数据的问答系统

GitHub代码仓库 架构 从高层次来看,这些系统的步骤如下: 将问题转换为SQL查询:模型将用户输入转换为SQL查询。 执行SQL查询:执行查询。 回答问题:模型根据查询结果响应用户输入。 样本数据 下载样本数据&#xf…

AI学习之-阿里天池

阿里天池(Tianchi)是阿里巴巴集团旗下的一个数据科学与人工智能竞赛平台,致力于推动数据科学和人工智能的发展。在天池平台上,人们可以参与各种数据竞赛和挑战,解决实际问题,提升数据科学技能。天池平台提供…

数据库管理-第295期 IT架构与爆炸半径(20250221)

数据库管理295期 2025-02-21 数据库管理-第295期 架构与爆炸半径(20250221)1 术语新解2 硬件:存储VS本地盘3 数据库3.1 多模VS专用3.2 集中式VS分布式 4 公有云VS非公有云总结 数据库管理-第295期 架构与爆炸半径(20250221&#x…