当前位置:首页 > 科技  > 软件

Java编程中必知必会的五条SOLID原则

来源: 责编: 时间:2023-10-10 18:31:27 169观看
导读在面向对象编程(OOP)领域,SOLID原则是类设计的指导准则。这五个原则形成了一套规则和最佳实践,开发人员在设计类结构时应遵循这些原则。通过理解和应用这些原则,我们可以发挥出设计模式的潜力,创建强大的软件架构。在本文中

在面向对象编程(OOP)领域,SOLID原则是类设计的指导准则。这五个原则形成了一套规则和最佳实践,开发人员在设计类结构时应遵循这些原则。通过理解和应用这些原则,我们可以发挥出设计模式的潜力,创建强大的软件架构。lRf28资讯网——每日最新资讯28at.com

在本文中,我们介绍SOLID原则的核心内容,并帮助您理解这些原则如何应用到项目中。lRf28资讯网——每日最新资讯28at.com

lRf28资讯网——每日最新资讯28at.com

核心目标:编写可理解、无回归和可测试的代码

SOLID原则的核心目标是构建可理解、无回归、可读和可测试的代码。这些代码作为一个画布,多个开发人员可以轻松地进行协作,促进生产力和创新的环境。lRf28资讯网——每日最新资讯28at.com

现在,我们逐个深入讨论SOLID原则,揭示它们在塑造类设计中的重要性。lRf28资讯网——每日最新资讯28at.com

  • S — 单一责任原则
  • O — 开放封闭原则
  • L — 里氏替换原则
  • I — 接口隔离原则
  • D — 依赖反转原则

1. 单一责任原则

SOLID的第一个支柱,单一责任原则(SRP),强调一个类应该只有一个变化的原因。遵循这个原则,我们确保每个类只负责一个任务,可使将来的维护和修改更容易。lRf28资讯网——每日最新资讯28at.com

/*手机类,具有属性和构造函数*/Class MobilePhone{    String brandName;    Float price;    Date manufactureDate;  public MobilePhone(String brandName,Float price,Date manufactureDate){      this.brandName=brandName;      this.price=price;      this.manufactureDate=manufactureDate;  }};/*Invoice类拥有一个手机(mobile phone)和该手机的数量(quantity)*/Class Invoice{    private MobilePhone mPhone;    int quantity;  public Invoice(MobilePhone mPhone,quantity){    this.mPhone=mPhone;    this.quantity=quantity;  }  public float calculateTotalPrice(){    return mPhone.price*this.quantity;//返回发票的总金额  }  public void printInvoice(){    //打印发票的逻辑  }  public void sendNotification(){    //发送通知的逻辑  }}

以上代码,与计算、打印或通知逻辑相关的任何更改都需要修改Invoice类。因此,发票类缺乏明确的关注点分离,对一个方面的修改会影响到其他功能。为了遵循SRP,关键是将Invoice类重构为更小、更专注的类, 每个类都独立处理特定的职责,比如计算、打印或通知逻辑。lRf28资讯网——每日最新资讯28at.com

根据职责将代码分离成独立的类是遵循单一责任原则(SRP)并促进可维护和灵活的代码库的正确方法。lRf28资讯网——每日最新资讯28at.com

在这种设计中,我们应该有:lRf28资讯网——每日最新资讯28at.com

  • Invoice类只包含计算逻辑。
  • InvoicePrint类只包含打印逻辑。
  • InvoiceNotify类只包含发送通知的逻辑。
/*手机类,具有属性和构造函数*/Class MobilePhone{    String brandName;    Float price;    Date manufactureDate;  public MobilePhone(String brandName,Float price,Date manufactureDate){      this.brandName=brandName;      this.price=price;      this.manufactureDate=manufactureDate;  }};/*发票类,拥有手机和数量属性*/Class Invoice{    private MobilePhone mPhone;    int quantity;  public Invoice(MobilePhone mPhone,quantity){    this.mPhone=mPhone;    this.quantity=quantity;  }  public float calculateTotalPrice(){    return mPhone.price*this.quantity;//返回发票的总金额  }}Class InvoicePrint{  private Invoice invoice;    public InvoicePrint(Invoice invoice){  this.invoice=invoice  }  public void printInvoice(){    //打印发票的逻辑  }}/*如果打印逻辑发生变化,只有InvoicePrint类会发生变化。*/Class InvoiceNotify{  private Invoice invoice;    public InvoiceNotify(Invoice invoice){  this.invoice=invoice  public void sendNotification(){    //发送通知给用户的逻辑  }}/*如果通知逻辑发生变化,只有InvoiceNotify类会发生变化。*/

2. 开放-封闭原则

第二个原则是开放-封闭原则(OCP),它鼓励软件实体对扩展开放,但对修改封闭。换句话说,一旦一个类被建立,它应该能够轻松地进行扩展,而不需要修改其现有的代码。这促进了代码的重用和稳定性。lRf28资讯网——每日最新资讯28at.com

让我们以上面使用的InvoiceNotify类为例,InvoiceNotify类经过测试,并且当前在客户端中实际使用,通过电子邮件发送发票通知。lRf28资讯网——每日最新资讯28at.com

现在有一个客户需求,他们需要通过推送通知发送通知。lRf28资讯网——每日最新资讯28at.com

Class InvoiceNotify{  private Invoice invoice;    public InvoiceNotify(Invoice invoice){  this.invoice=invoice  public void sendNotification(){    //发送通知给用户的逻辑  }  public void sendPushNotification(){    //发送推送通知给用户的逻辑  }}

以上代码,通过在现有类中添加一个新方法,我们违反了开放/封闭原则lRf28资讯网——每日最新资讯28at.com

与其在现有类中添加一个新方法,我们应该设计一个接口并在各个类中实现。lRf28资讯网——每日最新资讯28at.com

Interface InvoiceNotification{ public void sendNotification();}Class EmailNotification implements InvoiceNotification{ private Invoice invoice;  public EmailNotification(Invoice invoice){     this.invoice=invoice;  }  @Override  public void sendNotification(){  //通过电子邮件发送通知的逻辑  }}Class PushNotification implements InvoiceNotification{ private Invoice invoice;  public PushNotification(Invoice invoice){     this.invoice=invoice;  }  @Override  public void sendNotification(){  //发送推送通知的逻辑  }}

如果进一步增强需求,需要通过短信发送通知,无需修改现有类。相反,我们可以创建一个名为TextNotification的新类,它实现了InvoiceNotification接口并重写了sendNotification()方法。这样,我们就能够顺利地集成新功能,而不会破坏现有的代码库。lRf28资讯网——每日最新资讯28at.com

3. 里氏替换原则

里氏替换原则(LSP)定义了基类和派生类之间的契约。它规定派生类应该能够替代其基类,而不会影响程序的正确性。实质上,遵循这个原则可以确保继承被谨慎地使用,并保持类层次结构的完整性。lRf28资讯网——每日最新资讯28at.com

例如:在数学中,正方形可以被归类为矩形的一种特殊形式。它们之间的“是一个”关系可能会导致我们考虑在代码中使用继承来建模这种关系。然而,将正方形实现为矩形的派生类可能会导致意外的行为。lRf28资讯网——每日最新资讯28at.com

在数学中,正方形确实是矩形的一种特殊形式,正如“是一个”关系所暗示的那样。这往往会引诱我们在代码中使用继承来建模这种关系。然而,将正方形实现为矩形的派生类可能会导致意想不到和违反直觉的行为。lRf28资讯网——每日最新资讯28at.com

我们用一个简单的Java代码示例来说明这个问题:lRf28资讯网——每日最新资讯28at.com

class Rectangle {    protected int width;    protected int height;    public void setWidth(int width) {        this.width = width;    }    public void setHeight(int height) {        this.height = height;    }    public int calculateArea() {        return width * height;    }}class Square extends Rectangle {    @Override    public void setWidth(int width) {        this.width = width;        this.height = width; // 正方形的边长始终相等,所以两个维度都设置为相同的值。    }    @Override    public void setHeight(int height) {        this.height = height;        this.width = height; // 正方形的边长始终相等,所以两个维度都设置为相同的值。    }}public class Main {    public static void main(String[] args) {        Rectangle rectangle = new Square();        rectangle.setWidth(5);        rectangle.setHeight(3);        System.out.println("Area: " + rectangle.calculateArea());    }}

在这个例子中,我们有一个基类Rectangle,其中包含setWidth和setHeight方法,分别用于设置矩形的宽度和高度。Square类继承Rectangle类,并重写这些方法,以确保两个维度保持相等,以保持正方形的特性。lRf28资讯网——每日最新资讯28at.com

在主方法中,我们创建一个Rectangle引用,指向一个Square对象。当我们尝试为宽度和高度设置不同的值(分别为5和3)时,我们得到了一个边长为3的正方形,而不是实际宽度为5、高度为3的矩形。因此,计算得到的面积(9)与我们期望从宽度为5、高度为3的矩形得到的面积不符。lRf28资讯网——每日最新资讯28at.com

这个场景展示了里氏替换原则被违反的情况,通过Rectangle引用使用Square对象导致了意外的行为。lRf28资讯网——每日最新资讯28at.com

为了解决Square继承Rectangle的问题,我们需要重新评估继承关系和类设计。一种方法是在这种情况下避免使用继承,而是专注于公共接口或组合。我们用Java代码来说明解决方案:lRf28资讯网——每日最新资讯28at.com

interface Shape {    int calculateArea();}class Rectangle implements Shape {    protected int width;    protected int height;    public void setWidth(int width) {        this.width = width;    }    public void setHeight(int height) {        this.height = height;    }    @Override    public int calculateArea() {        return width * height;    }}class Square implements Shape {    protected int side;    public void setSide(int side) {        this.side = side;    }    @Override    public int calculateArea() {        return side * side;    }}public class Main {    public static void main(String[] args) {        Shape rectangle = new Rectangle();        rectangle.setWidth(5);        rectangle.setHeight(3);        System.out.println("矩形面积: " + rectangle.calculateArea());        Shape square = new Square();        square.setSide(5);        System.out.println("正方形面积: " + square.calculateArea());    }}

在这个解决方案中,我们引入了一个名为Shape的公共接口,定义了calculateArea()方法。现在,Rectangle和Square都实现了这个接口。Rectangle类保留了setWidth和setHeight方法,而Square类有一个setSide方法。每个类根据自己特定的属性计算面积。lRf28资讯网——每日最新资讯28at.com

现在,在main方法中,我们为Rectangle和Square对象分别创建了不同的Shape引用。我们可以适当设置尺寸而不会遇到任何问题。lRf28资讯网——每日最新资讯28at.com

通过使用组合和共同接口,我们确保每个形状都能独立运作,并且按预期运行,而不违反里氏替换原则。这种设计使我们能够优雅地处理不同的形状,促进了更清晰和可维护的代码库。lRf28资讯网——每日最新资讯28at.com

4. 接口隔离原则

接口隔离原则(ISP)建议客户端不应被强迫依赖于它们不使用的接口。与其拥有庞大而笨重的接口,更好的做法是创建小而专注的接口,以满足客户端的特定需求。lRf28资讯网——每日最新资讯28at.com

让我们通过一个简单的Java代码示例来说明ISP:lRf28资讯网——每日最新资讯28at.com

假设我们有一个名为Printer的接口,提供打印功能:lRf28资讯网——每日最新资讯28at.com

interface DocumentProcessor {    void print();    void fax();}class LaserPrinter implements DocumentProcessor {    @Override    public void print() {        System.out.println("Printing with a laser printer.");    }    @Override    public void fax() {        System.out.println("Sending a fax with a laser printer.");    }}class FaxMachine implements DocumentProcessor {    @Override    public void print() {        // 传真机无法打印,所以将这个方法保持为空。    }    @Override    public void fax() {        System.out.println("Sending a fax with a fax machine.");    }}

这个设计的问题在于FaxMachine类对于print()方法没有有意义的实现,因为传真机无法打印文档。尽管如此,FaxMachine类仍然被强制实现print()方法,这是因为DocumentProcessor接口的设计。lRf28资讯网——每日最新资讯28at.com

这种对接口隔离原则的违反显而易见,因为FaxMachine类现在需要实现它不需要或使用的方法。lRf28资讯网——每日最新资讯28at.com

5. 依赖反转原则

SOLID原则的最后一块拼图是依赖反转原则(Dependency Inversion Principle,DIP)。该原则主张高层模块不应依赖于低层模块,而应依赖于抽象。通过遵循这一原则,我们实现了解耦,从而增强了灵活性、可维护性和测试的便捷性。lRf28资讯网——每日最新资讯28at.com

让我们通过一个小的Java代码示例来说明违反依赖反转原则的情况:lRf28资讯网——每日最新资讯28at.com

假设我们有一个ReportGenerator类,它直接依赖于一个DatabaseConnection类:lRf28资讯网——每日最新资讯28at.com

class DatabaseConnection {    public void connect() {        System.out.println("Connected to the database.");    }    public void executeQuery(String query) {        System.out.println("Executing query: " + query);    }    public void close() {        System.out.println("Connection closed.");    }}class ReportGenerator {    private DatabaseConnection databaseConnection;    public ReportGenerator() {        this.databaseConnection = new DatabaseConnection();    }    public void generateReport() {        databaseConnection.connect();        databaseConnection.executeQuery("SELECT * FROM data_table");        databaseConnection.close();        System.out.println("Report generated successfully.");    }}

在这段代码中,ReportGenerator类在其构造函数中直接创建了一个DatabaseConnection实例。结果,ReportGenerator与DatabaseConnection紧密耦合。对DatabaseConnection类的任何更改都可能会影响到ReportGenerator。lRf28资讯网——每日最新资讯28at.com

为了解决这个问题,我们需要应用依赖反转原则,引入一个两个类都依赖的接口:lRf28资讯网——每日最新资讯28at.com

interface Connection {    void connect();    void executeQuery(String query);    void close();}class DatabaseConnection implements Connection {    @Override    public void connect() {        System.out.println("Connected to the database.");    }    @Override    public void executeQuery(String query) {        System.out.println("Executing query: " + query);    }    @Override    public void close() {        System.out.println("Connection closed.");    }}class ReportGenerator {    private Connection connection;    public ReportGenerator(Connection connection) {        this.connection = connection;    }    public void generateReport() {        connection.connect();        connection.executeQuery("SELECT * FROM data_table");        connection.close();        System.out.println("Report generated successfully.");    }}public class Main {    public static void main(String[] args) {        Connection databaseConnection = new DatabaseConnection();        ReportGenerator reportGenerator = new ReportGenerator(databaseConnection);        reportGenerator.generateReport();    }}

lRf28资讯网——每日最新资讯28at.com

通过遵循依赖反转原则,我们通过Connection接口解耦了ReportGenerator和DatabaseConnection类。这种方法允许我们在不修改ReportGenerator的情况下轻松切换和扩展Connection接口的实现。现在的代码符合原则,更易于维护和灵活。lRf28资讯网——每日最新资讯28at.com

结论

SOLID原则是面向对象类设计的基石,对于每个寻求创建高效、可维护和协作的软件的开发人员来说至关重要。当你踏上编码之旅时,请记住SOLID运用原则!lRf28资讯网——每日最新资讯28at.com

本文链接://www.dmpip.com//www.dmpip.com/showinfo-26-12720-0.htmlJava编程中必知必会的五条SOLID原则

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: Python 无敌?Kotlin 逆袭?TIOBE 9 月编程语言排行榜揭晓

下一篇: 学编程,为什么优先推荐学Python?

标签:
  • 热门焦点
  • 印度登月最关键一步!月船三号今晚进入环月轨道

    印度登月最关键一步!月船三号今晚进入环月轨道

    8月5日消息,据印度官方消息,月船三号将于北京时间今晚21时30分左右开始近月制动进入环月轨道。这是该探测器能够成功的最关键步骤之一,如果成功将开始围
  • 摸鱼心法第一章——和配置文件说拜拜

    摸鱼心法第一章——和配置文件说拜拜

    为了能摸鱼我们团队做了容器化,但是带来的问题是服务配置文件很麻烦,然后大家在群里进行了“亲切友好”的沟通图片图片图片图片对比就对比,简单对比下独立配置中心和k8s作为配
  • 十个简单但很有用的Python装饰器

    十个简单但很有用的Python装饰器

    装饰器(Decorators)是Python中一种强大而灵活的功能,用于修改或增强函数或类的行为。装饰器本质上是一个函数,它接受另一个函数或类作为参数,并返回一个新的函数或类。它们通常用
  • 19个 JavaScript 单行代码技巧,让你看起来像个专业人士

    19个 JavaScript 单行代码技巧,让你看起来像个专业人士

    今天这篇文章跟大家分享18个JS单行代码,你只需花几分钟时间,即可帮助您了解一些您可能不知道的 JS 知识,如果您已经知道了,就当作复习一下,古人云,温故而知新嘛。现在,我们就开始今
  • “又被陈思诚骗了”

    “又被陈思诚骗了”

    作者|张思齐 出品|众面(ID:ZhongMian_ZM)如今的国产悬疑电影,成了陈思诚的天下。最近大爆电影《消失的她》票房突破30亿断层夺魁暑期档,陈思诚再度风头无两。你可以说陈思诚的
  • 华为HarmonyOS 4.0将于8月4日发布 或搭载AI大模型技术

    华为HarmonyOS 4.0将于8月4日发布 或搭载AI大模型技术

    华为宣布HarmonyOS4.0将于8月4日正式发布。此前,华为已经针对开发者公布了HarmonyOS4.0,以便于开发者提前进行适配,也因此被曝光出了一些新系统的特性
  • 3699元!iQOO Neo8 Pro顶配版今日首销:1TB UFS 4.0同价位唯一

    3699元!iQOO Neo8 Pro顶配版今日首销:1TB UFS 4.0同价位唯一

    5月23日,iQOO推出了全新的iQOO Neo8系列,包含iQOO Neo8和iQOO Neo8 Pro两个版本,其中标准版搭载高通骁龙8+,而Pro版更是首发搭载了联发科天玑9200+旗舰
  • 回归OPPO两年,一加赢了销量,输了品牌

    回归OPPO两年,一加赢了销量,输了品牌

    成为OPPO旗下主打性能的先锋品牌后,一加屡创佳绩。今年618期间,一加手机全渠道销量同比增长362%,凭借一加 11、一加 Ace 2、一加 Ace 2V三款爆品,一加
  • 苹果140W USB-C充电器:采用氮化镓技术

    苹果140W USB-C充电器:采用氮化镓技术

    据10 月 30 日 9to5 Mac 消息报道,当苹果推出新的 MacBook Pro 2021 时,该公司还推出了新的 140W USB-C 充电器,附赠在 MacBook Pro 16 英寸机型的盒子里,也支
Top
Baidu
map