Java如何实现OOP设计原则?
时间:2026-03-13 12:40:39 388浏览 收藏
本文深入浅出地揭示了Java中“真正遵循面向对象设计原则”的本质——不是简单地使用class关键字,而是通过接口依赖、依赖注入和里氏替换等实践,让代码具备可替换、可扩展、低耦合的韧性:支付方式增删不改订单逻辑,模块改动互不影响,单元测试轻松隔离,新功能只需新增类而非修改旧代码;它把抽象的SOLID原则转化为开发者日常写new、if和public void时的一句自问:“以后加另一种实现,要动我几行?会破谁的假设?”——让面向对象回归人本,既好懂,又安全。

什么是“遵循面向对象设计原则”的实际表现
不是写个 class 就算面向对象,而是让类与类之间的关系可替换、可扩展、不耦合。比如你改一个支付模块,不该牵连订单、用户、日志模块;加一种新支付方式(如 Apple Pay),不该动已有代码,只新增类即可。这背后是 SOLID 原则的落地,但别被术语吓住——它本质是「写人能轻松看懂、机器能安全扩改」的代码习惯。
用 interface 而不是 class 做依赖声明
常见错误:把具体实现类(如 AlipayService)直接写进业务逻辑里,导致后续加 WechatPayService 时必须改方法签名、if-else 判断、甚至单元测试全崩。
正确做法是先定义契约:
public interface PaymentService {
boolean pay(Order order, BigDecimal amount);
}
再让具体实现去实现它:
public class AlipayService implements PaymentService { ... }
public class WechatPayService implements PaymentService { ... }
业务类只依赖接口:
public class OrderService {
private final PaymentService paymentService; // 不是 AlipayService!
public OrderService(PaymentService paymentService) {
this.paymentService = paymentService;
}
public void checkout(Order order) {
paymentService.pay(order, order.getTotal());
}
}
- 测试时可用
Mockito.mock(PaymentService.class)快速隔离 - 切换支付渠道只需改构造注入的对象,不碰业务逻辑行
- 违反这点,
Open/Closed Principle(开闭原则)就塌了一半
避免在构造函数里做重操作或强依赖初始化
典型反例:new DatabaseConnection()、loadConfigFromFile()、initHttpClient() 直接塞进构造函数。结果一 new 对象就可能抛 IOException 或阻塞线程,且无法 mock、无法控制生命周期。
更稳妥的做法:
- 构造函数只接收已创建好的依赖(即依赖注入),保持轻量、无副作用
- 耗时/异常风险操作移到
init()方法,由调用方显式触发(如 Spring 的@PostConstruct) - 若必须懒加载,用
Supplier或Optional包一层,推迟实例化时机
例如:
public class ReportGenerator {
private final Supplier<PdfRenderer> rendererSupplier;
public ReportGenerator(Supplier<PdfRenderer> rendererSupplier) {
this.rendererSupplier = rendererSupplier; // 不立即 new
}
public byte[] generate() {
PdfRenderer renderer = rendererSupplier.get(); // 真正需要时才创建
return renderer.render(data);
}
}
子类不应破坏父类的行为契约
这是 Liskov Substitution Principle(里氏替换原则)最常踩的坑。比如父类 CollectionUtils.isEmpty(List> list) 规定:传 null 返回 true;结果子类重写后对 null 抛 NullPointerException,上游所有调用立刻崩溃。
判断是否合规,看三点:
- 子类方法的参数范围不能比父类更窄(比如父类接受
Object,子类不能只接受String) - 子类方法的返回值类型不能比父类更宽(比如父类返回
List,子类不能返回ArrayList) - 子类不能在父类没声明异常的地方抛新异常,也不能取消父类声明的异常处理义务
更简单的方法:只要父类能跑通的单元测试,换成子类实例也必须全部通过。
真正难的不是记住五条原则名字,而是每次写 new、写 if、写 public void 时,多问一句:“以后有人要加另一种实现,得动我几行?会破谁的假设?”
到这里,我们也就讲完了《Java如何实现OOP设计原则?》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
420 收藏
-
359 收藏
-
112 收藏
-
305 收藏
-
309 收藏
-
341 收藏
-
116 收藏
-
180 收藏
-
425 收藏
-
192 收藏
-
307 收藏
-
335 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习