0

0

使用ArchUnit强制执行单一依赖:服务与存储库的架构约束

聖光之護

聖光之護

发布时间:2025-11-09 14:38:00

|

765人浏览过

|

来源于php中文网

原创

使用ArchUnit强制执行单一依赖:服务与存储库的架构约束

本文将详细介绍如何使用archunit定义并强制执行一项架构规则:确保每个存储库(repository)类只能被一个服务(service)类所依赖。我们将探讨如何通过自定义archcondition来精确检查依赖数量,并生成清晰的违规消息,从而有效维护应用模块间的单一职责和解耦性。

理解架构约束:存储库的单一服务依赖

在许多分层架构中,服务层(Service Layer)和数据访问层(Repository Layer)是常见的组件。通常,一个服务可以依赖多个存储库来完成其业务逻辑,但为了保持架构的清晰性、可维护性和单一职责原则,有时需要强制规定一个存储库只能被一个特定的服务所使用,即不允许存储库在多个服务之间共享。这种约束有助于避免复杂的交叉依赖和潜在的副作用。

ArchUnit是一个强大的Java架构测试库,它允许开发者以代码的形式定义和验证架构规则。接下来,我们将探讨如何使用ArchUnit来实现“存储库只能被一个服务使用”这一特定的架构规则。

初步规则:确保存储库仅被服务层使用

在强制单一服务依赖之前,一个更基础的规则是确保存储库类仅被服务层中的类所依赖,而不是被其他层(如控制器层或工具类)直接依赖。这可以通过以下ArchUnit规则实现:

import com.tngtech.archunit.core.domain.JavaClasses;
import com.tngtech.archunit.junit.ArchTest;
import com.tngtech.archunit.lang.ArchRule;

import static com.tngtech.archunit.lang.syntax.ArchRuleDefinition.classes;

public class RepositoryServiceRules {

    // 假设这些是定义包名的常量
    private static final String SUBPACKAGE_NAME_REPOSITORY = "..repository..";
    private static final String SUBPACKAGE_NAME_SERVICE = "..service..";

    @ArchTest
    static final ArchRule repository_must_only_be_used_by_a_service =
            classes().that().resideInAnyPackage(SUBPACKAGE_NAME_REPOSITORY)
                    .should().onlyHaveDependentClassesThat()
                    .resideInAnyPackage(SUBPACKAGE_NAME_SERVICE);
}

这条规则定义了:所有位于 ..repository.. 包中的类,它们的所有依赖者(即使用它们的类)都必须位于 ..service.. 包中。这确保了存储库不会被服务层以外的组件直接访问。然而,这条规则并没有限制一个存储库可以被多少个服务使用,它仅仅确保了依赖者是服务。

强制单一服务依赖:使用自定义ArchCondition

为了实现“一个存储库只能被一个服务使用”的严格约束,我们需要检查每个存储库类的直接依赖者数量。ArchUnit提供了 ArchCondition 机制,允许我们定义复杂的自定义检查逻辑。

方案一:简洁的Lambda表达式与`describe`

ArchUnit允许使用 describe 方法结合Lambda表达式快速定义一个简单的 ArchCondition。这种方式适用于逻辑相对简单,且默认违规消息可以接受的情况。

Inworld.ai
Inworld.ai

InWorldAI是一个AI角色开发平台,开发者可以创建具有自然语言、上下文意识和多模态的AI角色,并可以继承到游戏和实时媒体中

下载
import com.tngtech.archunit.base.DescribedPredicate;
import com.tngtech.archunit.core.domain.Dependency;
import com.tngtech.archunit.core.domain.JavaClass;
import com.tngtech.archunit.junit.ArchTest;
import com.tngtech.archunit.lang.ArchRule;

import static com.tngtech.archunit.base.DescribedPredicate.describe;
import static com.tngtech.archunit.lang.conditions.ArchConditions.have;
import static com.tngtech.archunit.lang.syntax.ArchRuleDefinition.classes;

public class RepositoryServiceRules {

    private static final String SUBPACKAGE_NAME_REPOSITORY = "..repository..";
    private static final String SUBPACKAGE_NAME_SERVICE = "..service..";

    @ArchTest
    ArchRule repository_must_have_exactly_one_dependent_class =
        classes().that().resideInAnyPackage(SUBPACKAGE_NAME_REPOSITORY)
            .should(have(describe("#{dependent classes} == 1", javaClass ->
                javaClass.getDirectDependenciesToSelf().stream()
                    .map(Dependency::getOriginClass).count() == 1
            )));
}

在这段代码中:

  • classes().that().resideInAnyPackage(SUBPACKAGE_NAME_REPOSITORY) 选择了所有存储库类作为规则检查的对象。
  • .should(have(describe(...))) 应用了一个自定义条件。
  • describe("#{dependent classes} == 1", ...) 定义了一个描述性谓词。其中的字符串是当规则通过时显示的描述,而Lambda表达式 javaClass -> ... 则是实际的检查逻辑。
  • javaClass.getDirectDependenciesToSelf() 获取了所有直接依赖于当前 javaClass 的依赖关系。
  • .stream().map(Dependency::getOriginClass) 将这些依赖关系映射到它们的源类(即依赖者类)。
  • .count() == 1 检查依赖者的数量是否恰好为1。

这个方案简洁有效,但当规则被违反时,生成的错误消息可能不够详细,仅显示“dependent classes == 1”的条件未满足。

方案二:自定义ArchCondition以生成更友好的违规消息

为了提供更具体、更易于理解的违规消息,我们可以实现一个完整的 ArchCondition。这在大型项目或复杂规则中尤其有用,因为它能帮助开发者快速定位问题。

import com.tngtech.archunit.core.domain.Dependency;
import com.tngtech.archunit.core.domain.JavaClass;
import com.tngtech.archunit.junit.ArchTest;
import com.tngtech.archunit.lang.ArchCondition;
import com.tngtech.archunit.lang.ArchRule;
import com.tngtech.archunit.lang.ConditionEvents;

import java.util.Set;
import java.util.stream.Collectors;

import static com.tngtech.archunit.lang.ConditionEvent.createMessage;
import static com.tngtech.archunit.lang.SimpleConditionEvent.violated;
import static com.tngtech.archunit.lang.syntax.ArchRuleDefinition.classes;
import static java.util.stream.Collectors.joining;
import static java.util.stream.Collectors.toSet;

public class RepositoryServiceRules {

    private static final String SUBPACKAGE_NAME_REPOSITORY = "..repository..";
    private static final String SUBPACKAGE_NAME_SERVICE = "..service..";

    @ArchTest
    ArchRule repository_must_have_exactly_one_dependent_class_with_custom_message =
        classes().that().resideInAnyPackage(SUBPACKAGE_NAME_REPOSITORY)
            .should(new ArchCondition<JavaClass>("have one dependent class") {
                @Override
                public void check(JavaClass javaClass, ConditionEvents events) {
                    // 获取所有直接依赖于当前存储库类的类
                    Set<JavaClass> dependentClasses = 
                        javaClass.getDirectDependenciesToSelf().stream()
                            .map(Dependency::getOriginClass)
                            .collect(toSet());

                    // 检查依赖者数量是否不为1
                    if (dependentClasses.size() != 1) {
                        String message;
                        if (dependentClasses.isEmpty()) {
                            // 如果没有依赖者
                            message = "has no dependent classes";
                        } else {
                            // 如果有多个依赖者,列出它们
                            message = dependentClasses.stream()
                                .map(JavaClass::getName)
                                .collect(joining(", ", "has several dependent classes: ", ""));
                        }
                        // 报告违规,并附带详细消息
                        events.add(violated(javaClass, createMessage(javaClass, message)));
                    }
                }
            });
}

这个自定义 ArchCondition 的实现提供了以下优点:

  • 清晰的描述: new ArchCondition("have one dependent class") 定义了规则的意图。
  • check 方法: 这是核心逻辑所在,它接收 JavaClass(当前正在检查的类)和 ConditionEvents(用于报告违规)。
  • 获取依赖者: 同样使用 javaClass.getDirectDependenciesToSelf().stream().map(Dependency::getOriginClass).collect(toSet()) 来获取所有依赖于当前存储库类的集合。
  • 详细的违规消息:
    • 如果 dependentClasses 为空,则报告“has no dependent classes”。
    • 如果 dependentClasses 大于1,则列出所有依赖者的全限定名,例如“has several dependent classes: com.example.service.UserService, com.example.service.ProductService”。
  • violated 和 createMessage: 这些方法用于向 ConditionEvents 报告具体的违规事件和消息。

通过这种方式,当ArchUnit测试失败时,开发者可以立即看到是哪个存储库类违反了规则,以及具体被哪些服务类共享,从而大大提高了问题排查的效率。

注意事项与最佳实践

  1. 包名约定: 示例中使用了 SUBPACKAGE_NAME_REPOSITORY 和 SUBPACKAGE_NAME_SERVICE 等常量来定义包路径。在实际项目中,应确保这些包名能够准确地匹配您的项目结构,并且具有良好的命名约定。
  2. getDirectDependenciesToSelf() 的理解: 这个方法返回的是所有直接依赖于当前 JavaClass 的 Dependency 对象。每个 Dependency 对象包含了依赖的源(getOriginClass())和目标(getTargetClass()),以及依赖的类型。
  3. 测试覆盖: 除了编写这些ArchUnit规则,还应确保这些规则被集成到CI/CD流程中,以便在代码提交时自动验证架构合规性。
  4. 规则粒度: 根据项目需求,您可以调整规则的粒度。例如,如果某些存储库确实需要被多个服务共享(例如,通用的用户认证存储库),则可能需要为这些特殊情况定义豁免规则,或者将它们放置在不同的包中,以便不被此规则约束。
  5. 可读性: 编写清晰的ArchUnit规则描述和详细的违规消息对于团队协作和长期维护至关重要。

总结

通过ArchUnit的自定义 ArchCondition 机制,我们可以灵活且精确地定义复杂的架构规则,例如强制存储库只能被单一服务依赖。这不仅有助于在开发早期发现架构问题,还能在项目演进过程中持续维护架构的健康性。无论是使用简洁的 describe 方法,还是实现更详细的 ArchCondition 来生成丰富的违规消息,ArchUnit都为Java项目的架构治理提供了强大的工具。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1564

2023.10.24

counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

203

2023.11.20

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

760

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

220

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1564

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

649

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

1208

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

1184

2024.04.29

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

4

2026.03.10

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.3万人学习

C# 教程
C# 教程

共94课时 | 11.1万人学习

Java 教程
Java 教程

共578课时 | 80.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号