转载-Archunit的使用
Archunit的使用
注:开发的编辑器: Intellij Idea,JDK版本是JDK8
Archunit是什么,官网的英文介绍很好,建议阅读原文,"ArchUnit is a free, simple and extensible library for checking the architecture of your Java code using any plain Java unit test framework. That is, ArchUnit can check dependencies between packages and classes, layers and slices, check for cyclic dependencies and more"。
简单来说,它是代码格式、类之间的依赖关系检查工具。
使用介绍
进入官网
点击右上角的"User Gruide"
之后就可以看到它的英文教程文档。
1.pom.xml中加入依赖
List-1 最重要的是archunit-junit4依赖
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>com.tngtech.archunit</groupId>
<artifactId>archunit-junit4</artifactId>
<version>0.9.1</version>
<scope>test</scope>
</dependency>
2.添加规则
直接上代码了,如下List-2所示,按项目情况,修改自己的"@AnalyzeClasses的packages值"
import javax.persistence.Entity;
import com.tngtech.archunit.junit.AnalyzeClasses;
import com.tngtech.archunit.junit.ArchTest;
import com.tngtech.archunit.junit.ArchUnitRunner;
import com.tngtech.archunit.lang.ArchRule;
import org.junit.runner.RunWith;
import static com.tngtech.archunit.lang.syntax.ArchRuleDefinition.classes;
import static com.tngtech.archunit.lang.syntax.ArchRuleDefinition.noClasses;
import static com.tngtech.archunit.library.GeneralCodingRules.NO_CLASSES_SHOULD_ACCESS_STANDARD_STREAMS;
import static com.tngtech.archunit.library.GeneralCodingRules.NO_CLASSES_SHOULD_THROW_GENERIC_EXCEPTIONS;
import static com.tngtech.archunit.library.GeneralCodingRules.NO_CLASSES_SHOULD_USE_JAVA_UTIL_LOGGING;
/**
* @author dmj1161859184@126.com 2018-09-14 23:03
* @version 1.0
* @since 1.0
*/
@RunWith(ArchUnitRunner.class) // Remove this line for JUnit 5!!
@AnalyzeClasses(packages = "com.mjduan.project") //要扫描的package
public class ArchunitTest {
/** dao下的类应该以Dao结尾 */
@ArchTest
public static final ArchRule DAOs_must_reside_in_a_dao_package = classes()
.that()
.haveNameMatching(".*Dao")
.should()
.resideInAPackage("..dao..")
.as("DAOs should reside in a package '..dao..'");
/** controller下的类应该以Controller结尾 */
@ArchTest
public static final ArchRule CONTROLLERs_must_reside_in_a_dao_package = classes()
.that()
.haveNameMatching(".*Controller")
.should()
.resideInAPackage("..controller..")
.as("DAOs should reside in a package '..dao..'");
/** util下的类应该以Util结尾 */
@ArchTest
public static final ArchRule UTILs_must_reside_in_a_dao_package = classes()
.that()
.haveNameMatching(".*Util")
.should()
.resideInAPackage("..util..")
.as("DAOs should reside in a package '..dao..'");
/** 有Entity注解的类,应该在domain或者entity包下 */
@ArchTest
public static final ArchRule entities_must_reside_in_a_domain_package = classes()
.that()
.areAnnotatedWith(Entity.class)
.should()
.resideInAnyPackage("..domain..", "..entity..");
/** 接口类,应该以I开头 */
@ArchTest
public static final ArchRule interface_className_must_start_with_I = classes()
.that()
.areInterfaces()
.should()
.haveSimpleNameStartingWith("I")
.as("Interface should start with I");
/** service下的类,只能被controller下或者时service下的类访问 */
@ArchTest
public static final ArchRule services_should_only_be_accessed_by_controllers_or_other_services = classes()
.that()
.resideInAPackage("..service..")
.should()
.onlyBeAccessed()
.byAnyPackage("..controller..", "..service..");
/** impl下的类,不能是interface */
@ArchTest
public static final ArchRule interfaces_must_not_be_placed_in_implementation_packages = noClasses()
.that()
.resideInAPackage("..impl..")
.should()
.beInterfaces();
/** 接口类,类名不能以Interface结尾 */
@ArchTest
public static final ArchRule interfaces_should_not_have_names_ending_with_the_word_interface = noClasses()
.that()
.areInterfaces()
.should()
.haveNameMatching(".*Interface");
/** service下的类,不能调用controller下的类 */
@ArchTest
public static final ArchRule services_should_not_access_controllers = noClasses()
.that()
.resideInAPackage("..service..")
.should()
.accessClassesThat()
.resideInAPackage("..controller..");
/** dao下的类,不能调用controller下的类 */
@ArchTest
public static final ArchRule controllers_should_not_access_dao = noClasses()
.that()
.resideInAPackage("..dao..")
.should()
.accessClassesThat()
.resideInAnyPackage("..controller..");
/** 不应该使用System.* */
@ArchTest
private final ArchRule NO_ACCESS_TO_STANDARD_STREAMS = NO_CLASSES_SHOULD_ACCESS_STANDARD_STREAMS;
/** 不应该抛出Exception */
@ArchTest
private final ArchRule NO_GENERIC_EXCEPTIONS = NO_CLASSES_SHOULD_THROW_GENERIC_EXCEPTIONS;
/** 不应该使用java.util.logging来记录日志 */
@ArchTest
private final ArchRule NO_JAVA_UTIL_LOGGING = NO_CLASSES_SHOULD_USE_JAVA_UTIL_LOGGING;
}
List-2中的代码中已经给出注释,自己可以在项目中运行看下结果。不过发现有个不好的地方是,archunit会扫描test下的类。
注:如果要忽略某个规则,那么加上@ArchIgnore就可以了,参考这里。
2.1 忽略某个规则
public class ArchitectureTest {
// will run
@ArchTest
public static final ArchRule rule1 = classes().should()...
// won't run
@ArchIgnore
@ArchTest
public static final ArchRule rule2 = classes().should()...
}
2.2 忽略多个类,即不让Archunit扫描多个类
一般出现在遗留系统中。
如图 图2.1

比如我们想在Archunit扫描时,忽略DemoInterface和AnnotationDemo,那么在resources下建个"archunit_ignore_patterns.txt"的文件(这个文件名称是固定的,不能修改),在将要忽略的类路径放入其中,注意格式,如下图2.2所示,参考其官网。
图2.2
3.Demo
新建一个interface,名为DemoInterface,之后运行List-2,结果如下图1所示,由于DemoInterface是interface,类名以Interface结尾(List-2中定义不能以Interface结尾),未以I开头(List-2中定义要以I开头):
图3.1 运行List-2后报错信息
图3.1 运行List-2后报错信息
4.官方的Archunit example
可以在Github上看Archunit的例子,Github地址: https://github.com/TNG/ArchUnit-Examples
实际项目中,代码量较多,规范要做好,特别是人员流动是公司中很常见的。
```
转载-Archunit的使用的更多相关文章
- Crystal Clear Applied: The Seven Properties of Running an Agile Project (转载)
作者Alistair Cockburn, Crystal Clear的7个成功要素,写得挺好. 敏捷方法的关注点,大家可以参考,太激动所以转载了. 原文:http://www.informit.com ...
- RTP与RTCP协议介绍(转载)
RTSP发起/终结流媒体.RTP传输流媒体数据 .RTCP对RTP进行控制,同步.RTP中没有连接的概念,本身并不能为按序传输数据包提供可靠的保证,也不提供流量控制和拥塞控制,这些都由RTCP来负责完 ...
- 《Walking the callstack(转载)》
本文转载自:https://www.codeproject.com/articles/11132/walking-the-callstack Download demo project with so ...
- [转载]MVVM模式原理分析及实践
没有找到很好的MVVM模式介绍文章,简单找了一篇,分享一下.MVVM实现了UI\UE设计师(Expression Blend 4设计界面)和软件工程师的合理分工,在SilverLight.WPF.Wi ...
- [转载]:STM32为什么必须先配置时钟再配置GPIO
转载来源 :http://blog.csdn.net/fushiqianxun/article/details/7926442 [原创]:我来添两句,就是很多同学(包括我)之前搞低端单片机,到了stm ...
- [转载]从MyEclipse到IntelliJ IDEA-让你摆脱鼠标,全键盘操作
从MyEclipse转战到IntelliJ IDEA的经历 注转载址:http://blog.csdn.net/luoweifu/article/details/13985835 我一个朋友写了一篇“ ...
- TCP同步与异步,长连接与短连接【转载】
原文地址:TCP同步与异步,长连接与短连接作者:1984346023 [转载说明:http://zjj1211.blog.51cto.com/1812544/373896 这是今天看到的一篇讲到T ...
- 在CentOS 7/6.5/6.4 中安装Java JDK 8(转载)
转载在CentOS 7/6.5/6.4 中安装Java JDK 8 首先,在你的服务器上运行一下更新. yum update 然后,在您的系统上搜索,任何版本的已安装的JDK组件. rpm -qa | ...
- 用C#实现MD5的加密(转载)
方法一 首先,先简单介绍一下MD5 MD5的全称是message-digest algorithm 5(信息-摘要算法,在90年代初由mit laboratory for computer scien ...
随机推荐
- jvm调优、常用工具
ps -ef | grep java查出进程id jmap -heap ID 查出jvm配置信息 加入参数:打印Gc日志,分析 GC日志分析工具: GCeasy 降低minor gc 和 full g ...
- JSP请求是如何被处理的?jsp的执行原理
客户端通过浏览器发送jsp请求,服务器端接受到请求后,判断是否是第一次请求该页面,或者该页面是否改变,若是,服务器将jsp页面翻译为servlet,jvm将servlet编译为.class文件,字节码 ...
- [修仙之路]React-Redux 金丹篇
作者:水涛追求技术,但又不失生活的乐趣,过自己想要的生活 React-Redux简介 React-Redux可以使你的React项目拥有全局数据,可以使多个React组件读取到全局数据并且组件中也可修 ...
- 常见面试题之*args
这个地方理解即可,只是面试的时候会被问到,单独做了一下知识点的整理,不推荐使用. def self_max(a,b,c,d,e,f,g,h,k,x=1,y=3,z=4): #默认参数 print(a, ...
- Nginx安装(我觉得我这篇可能是全网最清晰的一篇安装步骤了)
原文内容来自于LZ(楼主)的印象笔记,如出现排版异常或图片丢失等问题,可查看当前链接:https://app.yinxiang.com/shard/s17/nl/19391737/46aadb8f-5 ...
- 【高可用架构】用Nginx实现负载均衡(三)
前言 在上一篇,已经用Envoy工具统一发布了Deploy项目代码.本篇我们来看看如何用nginx实现负载均衡 负载均衡器IP 192.168.10.11 [高可用架构]系列链接:待部署的架构介绍 演 ...
- iOS编译自动升级版本号脚本
版权申明: 本文原创首发于以下网站,您可以自由转载,但必须加入完整的版权声明 博客园:https://www.cnblogs.com/MogooStudio/ csdn博客:https://blog. ...
- 编译安装基于 fastcgi 模式的多虚拟主机的wordpress和discuz的LAMP架构
目录 实现CentOS 7 编译安装基于 fastcgi 模式的多虚拟主机的wordpress和discuz的LAMP架构 准备环境: 准备软件版本: 主机名修改用以区分 数据库服务器 实现数据库二进 ...
- Django的Form验证
Django的Form验证 Form验证:Form提交Form表单数据验证 针对Form提交的数据进行验证 创建模板 class loginForm() 请求提交给模板,创建对象 obj=loginF ...
- 又到了抵制Notepad++的时候了?
逛开源中国(OSCHINA),无意中发现一贴<不用Notepad++,你还有这些更好的选择> 才发现,原来 Notepad++ 的作者侯今吾前几天又在 npp 的官网上发表了一篇个人政治意 ...