如何利用Java继承和多态性优化重复的验证逻辑

如何利用Java继承和多态性优化重复的验证逻辑

本文旨在解决Java中因处理不同但结构相似的对象类型而导致的验证逻辑重复问题。通过引入抽象父类和多态性,可以有效消除冗余代码,实现统一的验证方法,从而提高代码的可维护性、可扩展性,并遵循DRY(Don’t Repeat Yourself)原则。

消除Java中相似类型验证逻辑重复的策略

软件开发中,我们经常会遇到需要对结构相似但类型不同的数据对象执行相同或大部分相同的操作(例如验证)的场景。一个典型的例子是处理创建请求(createobjectrequest)和更新请求(updateobjectrequest),它们可能共享许多相同的字段,并且需要经过相同的验证规则。然而,如果为每种类型都编写一个独立的验证方法,会导致大量的代码重复,降低可维护性。

问题场景描述

考虑以下两个Java record 类型,它们分别用于创建和更新操作:

public record CreateObjectRequest (    CustomObjectA a,    CustomObjectB b,    CustomObjectC c) {}public record UpdateObjectRequest (    CustomObjectA a,    CustomObjectB b) {}

如果我们需要对这两种请求执行相同的验证逻辑,一个直观但效率低下的做法是为每种类型创建一个重载的验证方法:

public class RequestValidator {    public void validateRequest(CreateObjectRequest createObjectRequest) {        // 大量的验证逻辑,例如:        // if (createObjectRequest.a() == null) { throw new IllegalArgumentException("A cannot be null"); }        // if (createObjectRequest.b() == null) { throw new IllegalArgumentException("B cannot be null"); }        // ...    }    public void validateRequest(UpdateObjectRequest updateObjectRequest) {        // 与上述方法几乎相同的验证逻辑        // if (updateObjectRequest.a() == null) { throw new IllegalArgumentException("A cannot be null"); }        // if (updateObjectRequest.b() == null) { throw new IllegalArgumentException("B cannot be null"); }        // ...    }}

这种方法导致了显著的代码重复(”long body”),使得修改验证规则时需要同时修改多个地方,增加了出错的风险和维护成本。

解决方案:利用继承和多态性

为了解决这种代码重复问题,我们可以利用Java的继承和多态性特性。核心思想是识别出不同请求对象之间的共同点,并将这些共同点抽象到一个父类中。

立即学习“Java免费学习笔记(深入)”;

创建抽象父类

首先,定义一个抽象父类,例如 ObjectRequest,它包含所有子类共享的字段。请注意,Java record 类型不能直接继承其他类(除了 java.lang.Record),因此,为了实现继承,我们需要将 record 类型转换为传统的 class 类型。

public abstract class ObjectRequest {    protected CustomObjectA a;    protected CustomObjectB b;    // 构造函数、getter方法等(如果需要)    public ObjectRequest(CustomObjectA a, CustomObjectB b) {        this.a = a;        this.b = b;    }    public CustomObjectA getA() {        return a;    }    public CustomObjectB getB() {        return b;    }}

子类继承抽象父类

然后,让 CreateObjectRequest 和 UpdateObjectRequest 继承这个抽象父类,并添加它们各自特有的字段。

public class CreateObjectRequest extends ObjectRequest {    private CustomObjectC c; // CreateObjectRequest 特有的字段    public CreateObjectRequest(CustomObjectA a, CustomObjectB b, CustomObjectC c) {        super(a, b);        this.c = c;    }    public CustomObjectC getC() {        return c;    }    // 其他特定于CreateObjectRequest的方法}public class UpdateObjectRequest extends ObjectRequest {    // UpdateObjectRequest 可能没有特有字段,或者有其他特定字段    public UpdateObjectRequest(CustomObjectA a, CustomObjectB b) {        super(a, b);    }    // 其他特定于UpdateObjectRequest的方法}

统一验证方法

最后,将验证方法修改为接受抽象父类 ObjectRequest 作为参数。这样,无论是 CreateObjectRequest 还是 UpdateObjectRequest 的实例,都可以作为 ObjectRequest 类型传递给同一个验证方法。

public class RequestValidator {    public void validateRequest(ObjectRequest objectRequest) {        // 统一的验证逻辑        if (objectRequest.getA() == null) {            throw new IllegalArgumentException("Field A cannot be null.");        }        if (objectRequest.getB() == null) {            throw new IllegalArgumentException("Field B cannot be null.");        }        // 如果需要验证子类特有的字段,可以使用 instanceof 进行类型检查和向下转型        if (objectRequest instanceof CreateObjectRequest createRequest) {            if (createRequest.getC() == null) {                throw new IllegalArgumentException("Field C cannot be null for Create request.");            }        }        // 更多验证逻辑...    }}

优点与注意事项

减少代码重复(DRY原则):核心验证逻辑只需编写一次,极大地减少了冗余代码。提高可维护性:当验证规则发生变化时,只需修改一个地方。增强可扩展性:如果将来引入新的请求类型(如 DeleteObjectRequest),只要它继承 ObjectRequest,就可以直接使用现有的验证方法,或者只添加少量特定验证逻辑。清晰的类型层级:通过继承,明确了不同请求对象之间的“is-a”关系,提高了代码的可读性和结构性。

注意事项:

Record类型限制:如前所述,Java record 类型不能直接继承普通类。如果必须使用 record 类型,可以考虑使用接口(Interface)来定义共同的行为,但这通常不适用于共享字段的场景。对于共享字段和行为,将 record 转换为 class 是更直接的解决方案。字段访问修饰符:在父类中,如果子类需要直接访问父类的字段,可以使用 protected 修饰符。否则,通过 getter 方法访问是更好的实践。子类特定验证:如果子类有自己特有的字段或需要额外的验证规则,可以在统一验证方法内部使用 instanceof 运算符进行类型检查,并进行相应的向下转型来访问子类特有的属性和执行特定验证。

总结

通过将共同的字段和行为抽象到一个父类中,并利用Java的继承和多态性,我们可以有效地解决在处理结构相似但类型不同的对象时验证逻辑重复的问题。这种方法不仅减少了代码量,提高了代码质量,还使得系统更易于维护和扩展,是面向对象设计中的一个重要实践。

以上就是如何利用Java继承和多态性优化重复的验证逻辑的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/67185.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月12日 10:38:19
下一篇 2025年11月12日 11:20:08

相关推荐

  • 异常替代方案:Herb Sutter的error_code实践框架

    异常替代方案error_code通过返回值报告错误,避免抛出异常。1. error_code将错误码与上下文分离,可同时返回结果和丰富错误信息;2. 其本质是包含数值和error_category的轻量对象,避免模块间冲突;3. 与直接返回错误码相比,更灵活且无需为错误预留返回空间;4. 与异常相比…

    2025年12月18日 好文分享
    000
  • 如何注释代码?使用//单行或/* */多行注释

    写代码时加注释是为了提高代码可读性,方便自己和他人理解。应在关键地方添加注释,单行注释(//)适合解释单行代码或变量作用,如说明逻辑目的、调试屏蔽代码;多行注释(/ /)适合完整说明函数用途、参数含义及注意事项,并可用于临时屏蔽代码段;注释应清晰实用,避免重复代码内容、不相关背景或过时信息,应说明“…

    2025年12月18日 好文分享
    000
  • 模板如何支持多返回类型 auto和decltype(auto)的用法区别

    decltype(auto)与auto的关键区别在于类型推导时是否保留表达式的引用性和cv限定符。1.auto通过表达式值推导类型但忽略引用和const/volatile修饰,如int x推导为int、const int cx也推导为int;2.decltype(auto)则完整保留表达式原始类型特…

    2025年12月18日 好文分享
    000
  • 稳定地址方案:指针在容器扩容时不失效的魔法

    要保证容器扩容时指针、迭代器、引用有效,核心方法是使用间接访问机制。1. 句柄模式通过维护句柄到索引的映射,在扩容时不改变句柄,仅更新映射关系;2. 使用索引代替直接指针,只要元素位置不变,索引有效;3. 采用std::list或std::deque,其元素在插入删除时除被删元素外其他指针仍有效;4…

    2025年12月18日 好文分享
    000
  • #define如何定义宏?定义标识符替换文本

    宏定义是c++/c++中通过#define为文本指定别名的预处理指令。它将标识符替换为指定文本,不参与类型检查,仅做简单替换。例如#define pi 3.4159将所有pi替换为3.14159。使用时需注意:1.运算优先级问题,如带参数宏应加括号避免错误;2.避免参数含自增等副作用操作;3.用于定…

    2025年12月18日 好文分享
    000
  • 安全整数运算:避免overflow导致的安全漏洞

    安全整数运算的核心在于确保运算过程中不会发生溢出,从而避免程序行为异常或被攻击。1. 使用编译器或语言内置功能进行溢出检查,如 c++++20 的 std::has_overflow 和 rust 的 checked_add 方法;2. 手动实现溢出检测逻辑,例如通过判断 a + b 工具辅助检测;…

    2025年12月18日 好文分享
    000
  • c++中|的意思 按位或运算符使用场景示例

    在c++++中,| 符号代表按位或运算符,用于逐位比较两个操作数的二进制表示,若其中一位为1,结果的那一位即为1。1) 设置标志位:使用 |= 运算符可以方便地管理多个状态。2) 合并位掩码:通过 | 运算符组合选项,并用 & 运算符检查选项是否被设置。 在C++中,| 符号代表按位或运算符…

    2025年12月18日
    000
  • c++中/是什么意思 除法与注释符号区分

    在c++++中,/符号主要用作除法运算符和单行注释的开始符号。1)作为除法运算符时,/用于整数和浮点数的除法运算。2)作为单行注释的开始符号时,//后的内容会被忽略。通过上下文和良好的代码风格,可以区分这两种用法。 在C++中,/符号的用途主要有两种:作为除法运算符和作为单行注释的开始符号。让我们深…

    2025年12月18日
    000
  • 怎样在C++中实现文件复制?

    在c++++中实现文件复制的步骤包括:1)使用std::ifstream和std::ofstream打开源文件和目标文件;2)通过缓冲区逐块读取和写入文件内容;3)处理文件操作中的错误和异常;4)考虑使用std::filesystem简化文件复制过程;5)优化缓冲区大小和考虑并发复制、进度报告及跨平…

    2025年12月18日
    000
  • c++中!=的意思 不等于运算符使用示例

    在c++++中,!=运算符用于判断两个值是否不相等,返回布尔值true或false。1) 在条件判断、循环控制和算法设计中使用!=非常有用。2) 对于浮点数比较,应使用epsilon值避免精度问题。3) !=运算符在字符串处理和指针操作中也广泛应用,如检查用户输入和遍历链表。 在C++中,!=运算符…

    2025年12月18日
    000
  • 如何实现C++中的字符串匹配算法?

    c++++中的字符串匹配算法包括暴力匹配、kmp算法、boyer-moore算法和rabin-karp算法。1. 暴力匹配简单但效率低,适用于小规模数据。2. kmp算法通过部分匹配表提高效率,适用于大规模文本匹配。3. boyer-moore算法通过坏字符和好后缀规则提升匹配速度,适用于大文本和长…

    2025年12月18日
    000
  • 怎样在C++中实现分页查询?

    c++++中实现分页查询可以通过以下步骤实现:1.定义数据结构,使用std::vector存储数据;2.实现paginate函数,计算起始和结束索引,从数据库提取数据;3.优化计算总页数,使用gettotalpages函数;4.添加安全检查,实现safepaginate函数,确保输入参数有效性。 在…

    2025年12月18日
    000
  • 如何实现C++中的variant访问?

    c++++中的variant可以通过两种主要方式访问:1. 使用std::get直接访问特定类型;2. 使用std::visit处理所有可能类型。std::get适合访问特定类型,但可能触发异常,而std::visit更灵活但需定义访客对象。 实现C++中的variant访问可以说是现代C++编程中…

    2025年12月18日
    000
  • 如何实现C++中的移动语义?

    移动语义在c++++中通过移动构造函数和移动赋值运算符实现,其重要性在于提高资源管理效率。1. 移动构造函数使用右值引用接管资源并置原对象无效。2. 移动赋值运算符转移资源并释放原有资源,确保非自我赋值。 移动语义在C++中是一个非常酷的特性,让我们来聊聊如何实现它,以及为什么它如此重要。 当我第一…

    2025年12月18日
    000
  • c++中==什么意思 等于运算符使用注意事项

    在c++++中,==是等于运算符,用于比较两个操作数是否相等。使用时需注意:1) 不同数据类型行为不同,基本类型直接比较数值,自定义类型需重载==运算符;2) 浮点数比较需考虑精度问题,使用阈值判断;3) 指针比较的是地址,需解引用比较内容;4) 字符串需使用std::string或strcmp;5…

    2025年12月18日
    000
  • 怎样在C++中禁用标准库?

    在c++++中禁用标准库可以通过编译时使用-nostdlib选项实现,但需要手动处理内存管理和输入输出等。1. 使用-nostdlib编译选项禁用标准库。2. 手动处理内存分配和输入输出,如使用系统调用。3. 适用于资源受限的嵌入式系统和高安全性环境,但增加了代码复杂度和降低了可移植性。 在C++中…

    2025年12月18日
    000
  • 怎样在C++中使用if constexpr?

    在c++++中使用if constexpr可以大大提升代码的灵活性和性能。1) 它允许在编译时进行条件分支选择,减少二进制文件大小并提高运行时性能。2) 只能在编译时已知的条件下使用,且分支中定义的变量在其他分支不可见。3) 在模板元编程中特别有用,实现类型安全的函数重载。 在C++中使用if co…

    2025年12月18日
    000
  • c++中的/是什么意思 除号与注释符号区别

    在 c++++ 中,/ 符号既可以作为除法运算符,也可以作为单行注释的开始符号。1) 作为除法运算符时,用于数值除法,如 int result = 10 / 2;。2) 作为单行注释的开始符号时,// 后的内容被忽略,如 // 这是一条注释。区分这两种用法需注意上下文和使用技巧。 在 C++ 中,/…

    2025年12月18日
    000
  • c++中‖是什么意思 双竖线逻辑或运算符解析

    在c++++中,‖符号代表逻辑或运算符,用于判断两个条件中的任意一个是否为真。1) 它允许根据多个条件做出决定。2) 具有短路求值特性,可提高代码效率。3) 优先级较低,需使用括号确保逻辑正确性。4) 使用时需注意代码可读性和逻辑清晰。 在C++中,‖符号代表的是逻辑或运算符,它在编程中扮演着重要的…

    2025年12月18日
    000
  • C++中的注释规范是什么?

    c++++中的注释规范包括单行注释和多行注释。1. 单行注释使用//,适合简短注释。2. 多行注释使用/ /,适用于详细解释。注释应简洁、与代码同步,避免过度注释,以提高代码可读性和可维护性。 关于C++中的注释规范,我可以告诉你,C++的注释规范主要包括单行注释和多行注释的使用,以及如何在代码中有…

    2025年12月18日
    000

发表回复

登录后才能评论
关注微信