当前位置: 首页 > ai >正文

移动应用开发的六大设计原则

在移动应用开发中,遵循设计原则能大幅提升代码的可维护性和扩展性。本文以一个简单的学生管理系统为例,解析六大核心设计原则的实践方法。

1. 单一职责原则

优点:

  • 提高可维护性:一个类只负责一项职责,代码的功能会更加清晰。当需求发生变化时,只需要修改与该职责相关的类,不会对其他不相关的功能产生影响,从而降低了维护的复杂度。例如在学生管理系统中,Student 类专门负责存储学生信息,StudentManager 类专门负责学生信息的管理操作,若要修改学生信息的存储方式,只需关注 Student 类即可。
  • 增强可扩展性:每个类的职责明确,当需要添加新功能时,可以很方便地创建新的类来承担相应的职责,而不会破坏原有的代码结构。比如要添加学生成绩管理功能,可创建一个新的 StudentGradeManager 类,而不影响现有的 StudentStudentManager 类。
// 学生类,只负责存储学生信息
class Student {private String id;private String name;public Student(String id, String name) {this.id = id;this.name = name;}public String getId() {return id;}public String getName() {return name;}
}// 学生管理类,只负责学生信息的管理
import java.util.ArrayList;
import java.util.List;class StudentManager {private List<Student> students = new ArrayList<>();public void addStudent(Student student) {students.add(student);}public void removeStudent(Student student) {students.remove(student);}public Student findStudentById(String id) {for (Student student : students) {if (student.getId().equals(id)) {return student;}}return null;}

在上述代码中,Student 类只负责存储学生的基本信息,而 StudentManager 类只负责学生信息的管理操作,如添加、删除和查询。这样,每个类的职责都非常明确,当需求发生变化时,只需要修改相应的类即可。

2. 开闭原则

开闭原则要求软件实体(类、模块、函数等)应该对扩展开放,对修改关闭。

优点:

  • 提高软件的可扩展性:软件实体(类、模块、函数等)可以通过增加新的代码来扩展功能,而不需要修改已有的代码。这使得软件在面对不断变化的需求时,能够更加灵活地进行功能扩展,适应业务的发展和变化。
  • 保持软件的稳定性:避免了对原有代码的修改,从而减少了因修改而可能导致的错误和风险,保持了软件系统的稳定性和可靠性。已经经过测试和验证的代码可以继续稳定运行,不会因为新功能的添加而受到影响。
// 学生信息展示接口
interface StudentDisplay {void display(Student student);
}// 控制台展示实现类
class ConsoleStudentDisplay implements StudentDisplay {@Overridepublic void display(Student student) {System.out.println("Student ID: " + student.getId() + ", Name: " + student.getName());}
}// 学生管理类中使用接口
class StudentManager {private List<Student> students = new ArrayList<>();public void addStudent(Student student) {students.add(student);}public void removeStudent(Student student) {students.remove(student);}public Student findStudentById(String id) {for (Student student : students) {if (student.getId().equals(id)) {return student;}}return null;}public void displayStudent(String id, StudentDisplay display) {Student student = findStudentById(id);if (student != null) {display.display(student);}}
}

在上述代码中,我们定义了一个 StudentDisplay 接口,用于展示学生信息。ConsoleStudentDisplay 类实现了该接口,提供了在控制台展示学生信息的功能。当需要添加新的展示方式时,如将学生信息展示在文件中,我们只需要实现 StudentDisplay 接口,而不需要修改 StudentManager 类的代码,从而实现了对扩展开放,对修改关闭。

3. 里氏替换原则

里氏替换原则,即子类可以替换其父类并且不会影响程序的正确性。

优点:

  • 保证系统的稳定性:子类可以替换父类,使得在使用父类的地方可以透明地使用子类对象,不会影响程序的正确性。这保证了系统在扩展和维护过程中的稳定性,避免了因子类替换父类而导致的程序崩溃或错误。例如在学生管理系统中,NormalStudentExcellentStudent 类都可以替换 AbstractStudent 类,而不会影响 StudentManager 类的正常运行。
  • 促进代码复用:通过继承和多态,父类的代码可以被子类复用,减少了代码的重复编写。子类可以在父类的基础上进行扩展,实现自己的独特功能,提高了代码的复用率和开发效率。
  • 便于系统的扩展和维护:遵循里氏替换原则,使得系统的设计更加灵活,当需要添加新的子类时,不会对现有的代码产生影响,便于系统的扩展和维护。
// 抽象类
abstract class AbstractStudent {protected String id;protected String name;public AbstractStudent(String id, String name) {this.id = id;this.name = name;}public abstract String getInfo();
}// 普通学生类
class NormalStudent extends AbstractStudent {public NormalStudent(String id, String name) {super(id, name);}@Overridepublic String getInfo() {return "Normal Student - ID: " + id + ", Name: " + name;}
}// 优秀学生类
class ExcellentStudent extends AbstractStudent {public ExcellentStudent(String id, String name) {super(id, name);}@Overridepublic String getInfo() {return "Excellent Student - ID: " + id + ", Name: " + name;}
}// 学生管理类使用抽象学生类
class StudentManager {private List<AbstractStudent> students = new ArrayList<>();public void addStudent(AbstractStudent student) {students.add(student);}public void displayStudents() {for (AbstractStudent student : students) {System.out.println(student.getInfo());}}
}

在上述代码中,NormalStudentExcellentStudent 类都继承自 AbstractStudent 类。StudentManager 类使用 AbstractStudent 类来管理学生信息,这样无论是 NormalStudent 还是 ExcellentStudent 对象都可以替换 AbstractStudent 对象,而不会影响程序的正确性。

4. 依赖倒置原则

依赖倒置原则要求高层模块不应该依赖低层模块,二者都应该依赖抽象。

优点:

  • 降低模块间的耦合度:高层模块不依赖低层模块,二者都依赖抽象,减少了模块之间的直接依赖关系。当低层模块发生变化时,不会影响到高层模块,提高了系统的独立性和可维护性。例如在学生管理系统中,StudentViewer 类依赖 StudentInfoProvider 接口,而不是具体的 StudentInfoProviderImpl 类,当需要更换学生信息的获取方式时,只需实现新的 StudentInfoProvider 接口实现类,而无需修改 StudentViewer 类。
  • 提高系统的可测试性:依赖抽象使得可以更容易地对模块进行单元测试。可以使用模拟对象来替代具体的实现类,对模块进行独立测试,提高了测试的效率和准确性。
  • 增强系统的可扩展性:通过依赖抽象,系统可以方便地引入新的实现类,扩展系统的功能。例如在学生管理系统中,当需要添加新的学生信息获取方式时,只需实现 StudentInfoProvider 接口,而无需修改现有的代码。
// 学生信息获取接口
interface StudentInfoProvider {Student getStudentById(String id);
}// 学生信息获取实现类
class StudentInfoProviderImpl implements StudentInfoProvider {private List<Student> students = new ArrayList<>();public StudentInfoProviderImpl(List<Student> students) {this.students = students;}@Overridepublic Student getStudentById(String id) {for (Student student : students) {if (student.getId().equals(id)) {return student;}}return null;}
}// 学生信息展示依赖接口
class StudentViewer {private StudentInfoProvider provider;public StudentViewer(StudentInfoProvider provider) {this.provider = provider;}public void viewStudent(String id) {Student student = provider.getStudentById(id);if (student != null) {System.out.println("Student ID: " + student.getId() + ", Name: " + student.getName());}}
}

在上述代码中,StudentViewer 类是高层模块,StudentInfoProviderImpl 类是低层模块。StudentViewer 类不直接依赖 StudentInfoProviderImpl 类,而是依赖 StudentInfoProvider 接口。通过依赖注入的方式,将 StudentInfoProvider 接口的实现类注入到 StudentViewer 类中,从而实现了依赖倒置。

5. 接口隔离原则

接口隔离原则强调客户端不应该依赖它不需要的接口。

优点:

  • 提高系统的内聚性:每个接口只包含客户端需要的方法,使得接口的职责更加单一,提高了系统的内聚性。内聚性高的系统更容易维护和扩展。
  • 增强代码的灵活性:当需求发生变化时,只需要修改相关的接口和实现类,不会影响到其他不相关的接口和类,提高了代码的灵活性和可维护性。
// 学生添加接口
interface StudentAdder {void addStudent(Student student);
}// 学生查询接口
interface StudentFinder {Student findStudentById(String id);
}// 学生管理类实现多个接口
class StudentManager implements StudentAdder, StudentFinder {private List<Student> students = new ArrayList<>();@Overridepublic void addStudent(Student student) {students.add(student);}@Overridepublic Student findStudentById(String id) {for (Student student : students) {if (student.getId().equals(id)) {return student;}}return null;}
}

在上述代码中,我们将学生管理的功能拆分成了 StudentAdderStudentFinder 两个接口。StudentManager 类实现了这两个接口,这样不同的客户端可以根据自己的需求只依赖它们需要的接口,避免了依赖不必要的方法。

6. 迪米特法则

迪米特法则也称为最少知识原则,它要求一个对象应该对其他对象有最少的了解。

优点:

  • 降低系统的耦合度:一个对象对其他对象有最少的了解,减少了对象之间的直接交互,降低了系统的耦合度。当一个对象发生变化时,不会对其他对象产生过多的影响,提高了系统的独立性和可维护性。例如在学生管理系统中,StudentViewer 类只和 StudentManager 类进行交互,而不直接和 Student 类进行交互。
  • 提高系统的可维护性:由于对象之间的交互减少,系统的结构更加清晰,当需要修改某个对象时,只需要关注该对象及其直接关联的对象,降低了维护的难度。
  • 增强系统的安全性:减少对象之间的直接交互,避免了不必要的信息传递,提高了系统的安全性。例如在学生管理系统中,StudentViewer 类只能通过 StudentManager 类获取学生信息,不能直接访问 Student 类的内部信息,保护了学生信息的安全。
// 学生类
class Student {private String id;private String name;public Student(String id, String name) {this.id = id;this.name = name;}public String getId() {return id;}public String getName() {return name;}
}// 学生管理类
class StudentManager {private List<Student> students = new ArrayList<>();public void addStudent(Student student) {students.add(student);}public void removeStudent(Student student) {students.remove(student);}public Student findStudentById(String id) {for (Student student : students) {if (student.getId().equals(id)) {return student;}}return null;}
}// 学生信息页面
class StudentViewer {private StudentManager manager;public StudentViewer(StudentManager manager) {this.manager = manager;}public void viewStudent(String id) {Student student = manager.findStudentById(id);if (student != null) {System.out.println("Student ID: " + student.getId() + ", Name: " + student.getName());}}
}

在上述代码中,StudentViewer 类只和 StudentManager 类进行交互,而不直接和 Student 类进行交互。这样,StudentViewer 类对其他对象的了解最少,符合迪米特法则。

总结建议


代码是写给人看的,好的设计让应用既能快速响应需求变化,又能保持代码的优雅整洁。

http://www.xdnf.cn/news/5351.html

相关文章:

  • [Java实战]Spring Boot 整合 Freemarker (十一)
  • C++入门小馆: 二叉搜索树
  • 前端面试2
  • 【C语言干货】二维数组传参本质
  • C++23 views::repeat (P2474R2) 写一篇博客
  • Flutter - UIKit开发相关指南 - 导航
  • 深入理解 Java 适配器模式:架构设计中的接口转换艺术
  • 集成灶十大品牌对比
  • Nodejs核心机制
  • 说说Redis的内存淘汰策略?
  • 超市销售管理系统 - 需求分析阶段报告
  • Fiori学习专题四十:单一控件
  • 汇编学习——iOS开发对arm64汇编的初步了解
  • Spring Boot项目(Vue3+ElementPlus+Axios+MyBatisPlus+Spring Boot前后端分离)
  • 微服务架构实战:从服务拆分到RestTemplate远程调用
  • DINOv2
  • Spring框架(一)
  • Spring AI(3)——Chat Memory
  • skopeo工具详解
  • 成功案例:塔能精准节能技术为核心的工厂节能
  • GitHub打开缓慢甚至失败的解决办法
  • RTOS优先级翻转
  • 论文解读:MP-SfM: Monocular Surface Priors for Robust Structure-from-Motion
  • 22.第二阶段x64游戏实战-分析周围对象类型
  • SHAP分析!Transformer-BiLSTM组合模型SHAP分析,模型可解释不在发愁!
  • 分享一个可以用GPT打标的傻瓜式SD图片打标工具——辣椒炒肉图片打标助手
  • 04.three官方示例+编辑器+AI快速学习webgl_animation_skinning_additive_blending
  • 基于VSCode+PlatformIO环境的ESP8266的HX1838红外模块
  • sql的性能分析
  • Linux | Uboot-Logo 修改文档(第十七天)