Java学习指南
  • Java 编程的逻辑
  • Java进阶
  • Java FrameWorks
  • 了解 USB Type-A,B,C 三大标准接口
  • 深入浅出DDD
  • 重构:改善既有代码的设计
  • 面试大纲
  • 云原生
    • 什么是无服务器(what is serverless)?
  • 博客
    • 深入分析Log4j 漏洞
  • 博客
    • Serverless之快速搭建Spring Boot应用
  • 博客
    • 使用 Prometheus + Grafana + Spring Boot Actuator 监控应用
  • 博客
    • 使用 Prometheus + Grafana 监控 MySQL
  • 博客
    • 使用Github Actions + Docker 部署Spring Boot应用
  • 博客
    • Redis分布式锁之Redisson的原理和实践
  • 博客
    • 数据库中的树结构应该怎样去设计
  • 学习&成长
    • 如何成为技术大牛
  • 开发工具
    • Git Commit Message Guidelines
  • 开发工具
    • git命名大全
  • 开发工具
    • Gradle vs Maven Comparison
  • 开发工具
    • Swagger2常用注解及其说明
  • 开发工具
    • 简明 VIM 练级攻略
  • 微服务
    • 十大微服务设计模式和原则
  • 微服务
    • 微服务下的身份认证和令牌管理
  • 微服务
    • 微服务坏味道之循环依赖
  • 设计模式
    • 设计模式 - JDK中的设计模式
  • 设计模式
    • 设计模式 - Java三种代理模式
  • 设计模式
    • 设计模式 - 六大设计原则
  • 设计模式
    • 设计模式 - 单例模式
  • 设计模式
    • 设计模式 - 命名模式
  • 设计模式
    • 设计模式 - 备忘录模式
  • 设计模式
    • 设计模式 - 概览
  • 设计模式
    • 设计模式 - 没用的设计模式
  • 质量&效率
    • Homebrew 替换国内镜像源
  • 质量&效率
    • 工作中如何做好技术积累
  • Java FrameWorks
    • Logback
      • 自定义 logback 日志过滤器
  • Java FrameWorks
    • Mybatis
      • MyBatis(十三) - 整合Spring
  • Java FrameWorks
    • Mybatis
      • MyBatis(十二) - 一些API
  • Java FrameWorks
    • Mybatis
      • Mybatis(一) - 概述
  • Java FrameWorks
    • Mybatis
      • Mybatis(七) - 结果集的封装与映射
  • Java FrameWorks
    • Mybatis
      • Mybatis(三) - mapper.xml及其加载机制
  • Java FrameWorks
    • Mybatis
      • Mybatis(九) - 事务
  • Java FrameWorks
    • Mybatis
      • Mybatis(二) - 全局配置文件及其加载机制
  • Java FrameWorks
    • Mybatis
      • Mybatis(五) - SqlSession执行流程
  • Java FrameWorks
    • Mybatis
      • Mybatis(八) - 缓存
  • Java FrameWorks
    • Mybatis
      • Mybatis(六) - 动态SQL的参数绑定与执行
  • Java FrameWorks
    • Mybatis
      • Mybatis(十) - 插件
  • Java FrameWorks
    • Mybatis
      • Mybatis(十一) - 日志
  • Java FrameWorks
    • Mybatis
      • Mybatis(四) - Mapper接口解析
  • Java FrameWorks
    • Netty
      • Netty 可靠性分析
  • Java FrameWorks
    • Netty
      • Netty - Netty 线程模型
  • Java FrameWorks
    • Netty
      • Netty堆外内存泄露排查盛宴
  • Java FrameWorks
    • Netty
      • Netty高级 - 高性能之道
  • Java FrameWorks
    • Shiro
      • Shiro + JWT + Spring Boot Restful 简易教程
  • Java FrameWorks
    • Shiro
      • 非常详尽的 Shiro 架构解析!
  • Java FrameWorks
    • Spring
      • Spring AOP 使用介绍,从前世到今生
  • Java FrameWorks
    • Spring
      • Spring AOP 源码解析
  • Java FrameWorks
    • Spring
      • Spring Event 实现原理
  • Java FrameWorks
    • Spring
      • Spring Events
  • Java FrameWorks
    • Spring
      • Spring IOC容器源码分析
  • Java FrameWorks
    • Spring
      • Spring Integration简介
  • Java FrameWorks
    • Spring
      • Spring MVC 框架中拦截器 Interceptor 的使用方法
  • Java FrameWorks
    • Spring
      • Spring bean 解析、注册、实例化流程源码剖析
  • Java FrameWorks
    • Spring
      • Spring validation中@NotNull、@NotEmpty、@NotBlank的区别
  • Java FrameWorks
    • Spring
      • Spring 如何解决循环依赖?
  • Java FrameWorks
    • Spring
      • Spring 异步实现原理与实战分享
  • Java FrameWorks
    • Spring
      • Spring中的“for update”问题
  • Java FrameWorks
    • Spring
      • Spring中的设计模式
  • Java FrameWorks
    • Spring
      • Spring事务失效的 8 大原因
  • Java FrameWorks
    • Spring
      • Spring事务管理详解
  • Java FrameWorks
    • Spring
      • Spring计时器StopWatch使用
  • Java FrameWorks
    • Spring
      • 详述 Spring MVC 框架中拦截器 Interceptor 的使用方法
  • Java FrameWorks
    • Spring
      • 透彻的掌握 Spring 中@transactional 的使用
  • Java
    • Java IO&NIO&AIO
      • Java IO - BIO 详解
  • Java
    • Java IO&NIO&AIO
      • Java NIO - IO多路复用详解
  • Java
    • Java IO&NIO&AIO
      • Java N(A)IO - Netty
  • Java
    • Java IO&NIO&AIO
      • Java IO - Unix IO模型
  • Java
    • Java IO&NIO&AIO
      • Java IO - 分类
  • Java
    • Java IO&NIO&AIO
      • Java NIO - 基础详解
  • Java
    • Java IO&NIO&AIO
      • Java IO - 常见类使用
  • Java
    • Java IO&NIO&AIO
      • Java AIO - 异步IO详解
  • Java
    • Java IO&NIO&AIO
      • Java IO概述
  • Java
    • Java IO&NIO&AIO
      • Java IO - 设计模式
  • Java
    • Java IO&NIO&AIO
      • Java NIO - 零拷贝实现
  • Java
    • Java JVM
      • JVM 优化经验总结
  • Java
    • Java JVM
      • JVM 内存结构
  • Java
    • Java JVM
      • JVM参数设置
  • Java
    • Java JVM
      • Java 内存模型
  • Java
    • Java JVM
      • 从实际案例聊聊Java应用的GC优化
  • Java
    • Java JVM
      • Java 垃圾回收器G1详解
  • Java
    • Java JVM
      • 垃圾回收器Shenandoah GC详解
  • Java
    • Java JVM
      • 垃圾回收器ZGC详解
  • Java
    • Java JVM
      • 垃圾回收基础
  • Java
    • Java JVM
      • 如何优化Java GC
  • Java
    • Java JVM
      • 类加载机制
  • Java
    • Java JVM
      • 类字节码详解
  • Java
    • Java 基础
      • Java hashCode() 和 equals()
  • Java
    • Java 基础
      • Java 基础 - Java native方法以及JNI实践
  • Java
    • Java 基础
      • Java serialVersionUID 有什么作用?
  • Java
    • Java 基础
      • Java 泛型的类型擦除
  • Java
    • Java 基础
      • Java 基础 - Unsafe类解析
  • Java
    • Java 基础
      • Difference Between Statement and PreparedStatement
  • Java
    • Java 基础
      • Java 基础 - SPI机制详解
  • Java
    • Java 基础
      • Java 基础 - final
  • Java
    • Java 基础
      • Java中static关键字详解
  • Java
    • Java 基础
      • 为什么说Java中只有值传递?
  • Java
    • Java 基础
      • Java 基础 - 即时编译器原理解析及实践
  • Java
    • Java 基础
      • Java 基础 - 反射
  • Java
    • Java 基础
      • Java多态的面试题
  • Java
    • Java 基础
      • Java 基础 - 异常机制详解
  • Java
    • Java 基础
      • 为什么要有抽象类?
  • Java
    • Java 基础
      • 接口的本质
  • Java
    • Java 基础
      • Java 基础 - 枚举
  • Java
    • Java 基础
      • Java 基础 - 泛型机制详解
  • Java
    • Java 基础
      • Java 基础 - 注解机制详解
  • Java
    • Java 基础
      • 为什么 String hashCode 方法选择数字31作为乘子
  • Java
    • Java 并发
      • Java 并发 - 14个Java并发容器
  • Java
    • Java 并发
      • Java 并发 - AQS
  • Java
    • Java 并发
      • Java 并发 - BlockingQueue
  • Java
    • Java 并发
      • Java 并发 - CAS
  • Java
    • Java 并发
      • Java 并发 - Condition接口
  • Java
    • Java 并发
      • Java 并发 - CopyOnWriteArrayList
  • Java
    • Java 并发
      • Java 并发 - CountDownLatch、CyclicBarrier和Phaser对比
  • Java
    • Java 并发
      • Java 并发 - Fork&Join框架
  • Java
    • Java 并发
      • Java 并发 - Java CompletableFuture 详解
  • Java
    • Java 并发
      • Java 并发 - Java 线程池
  • Java
    • Java 并发
      • Java 并发 - Lock接口
  • Java
    • Java 并发
      • Java 并发 - ReentrantLock
  • Java
    • Java 并发
      • Java 并发 - ReentrantReadWriteLock
  • Java
    • Java 并发
      • Java 并发 - Synchronized
  • Java
    • Java 并发
      • Java 并发 - ThreadLocal 内存泄漏问题
  • Java
    • Java 并发
      • Java 并发 - ThreadLocal
  • Java
    • Java 并发
      • Java 并发 - Volatile
  • Java
    • Java 并发
      • Java 并发 - 从ReentrantLock的实现看AQS的原理及应用
  • Java
    • Java 并发
      • Java 并发 - 公平锁和非公平锁
  • Java
    • Java 并发
      • Java 并发 - 内存模型
  • Java
    • Java 并发
      • Java 并发 - 原子类
  • Java
    • Java 并发
      • Java 并发 - 如何确保三个线程顺序执行?
  • Java
    • Java 并发
      • Java 并发 - 锁
  • Java
    • Java 的新特性
      • Java 10 新特性概述
  • Java
    • Java 的新特性
      • Java 11 新特性概述
  • Java
    • Java 的新特性
      • Java 12 新特性概述
  • Java
    • Java 的新特性
      • Java 13 新特性概述
  • Java
    • Java 的新特性
      • Java 14 新特性概述
  • Java
    • Java 的新特性
      • Java 15 新特性概述
  • Java
    • Java 的新特性
      • Java 8的新特性
  • Java
    • Java 的新特性
      • Java 9 新特性概述
  • Java
    • Java 调试排错
      • 调试排错 - Java Debug Interface(JDI)详解
  • Java
    • Java 调试排错
      • 调试排错 - CPU 100% 排查优化实践
  • Java
    • Java 调试排错
      • 调试排错 - Java Heap Dump分析
  • Java
    • Java 调试排错
      • 调试排错 - Java Thread Dump分析
  • Java
    • Java 调试排错
      • 调试排错 - Java动态调试技术原理
  • Java
    • Java 调试排错
      • 调试排错 - Java应用在线调试Arthas
  • Java
    • Java 调试排错
      • 调试排错 - Java问题排查:工具单
  • Java
    • Java 调试排错
      • 调试排错 - 内存溢出与内存泄漏
  • Java
    • Java 调试排错
      • 调试排错 - 在线分析GC日志的网站GCeasy
  • Java
    • Java 调试排错
      • 调试排错 - 常见的GC问题分析与解决
  • Java
    • Java 集合
      • Java 集合 - ArrayList
  • Java
    • Java 集合
      • Java 集合 - HashMap 和 ConcurrentHashMap
  • Java
    • Java 集合
      • Java 集合 - HashMap的死循环问题
  • Java
    • Java 集合
      • Java 集合 - LinkedHashSet&Map
  • Java
    • Java 集合
      • Java 集合 - LinkedList
  • Java
    • Java 集合
      • Java 集合 - PriorityQueue
  • Java
    • Java 集合
      • Java 集合 - Stack & Queue
  • Java
    • Java 集合
      • Java 集合 - TreeSet & TreeMap
  • Java
    • Java 集合
      • Java 集合 - WeakHashMap
  • Java
    • Java 集合
      • Java 集合 - 为什么HashMap的容量是2的幂次方
  • Java
    • Java 集合
      • Java 集合 - 概览
  • Java
    • Java 集合
      • Java 集合 - 高性能队列Disruptor详解
  • 分布式
    • RPC
      • ⭐️RPC - Dubbo&hsf&Spring cloud的区别
  • 分布式
    • RPC
      • ⭐️RPC - Dubbo的架构原理
  • 分布式
    • RPC
      • ⭐️RPC - HSF的原理分析
  • 分布式
    • RPC
      • ⭐️RPC - 你应该知道的RPC原理
  • 分布式
    • RPC
      • ⭐️RPC - 动态代理
  • 分布式
    • RPC
      • 深入理解 RPC 之协议篇
  • 分布式
    • RPC
      • RPC - 序列化和反序列化
  • 分布式
    • RPC
      • ⭐️RPC - 服务注册与发现
  • 分布式
    • RPC
      • RPC - 核心原理
  • 分布式
    • RPC
      • ⭐️RPC - 框架对比
  • 分布式
    • RPC
      • ⭐️RPC - 网络通信
  • 分布式
    • 分布式事务
      • 分布式事务 Seata TCC 模式深度解析
  • 分布式
    • 分布式事务
      • 分布式事务的实现原理
  • 分布式
    • 分布式事务
      • 常用的分布式事务解决方案
  • 分布式
    • 分布式事务
      • 手写实现基于消息队列的分布式事务框架
  • 分布式
    • 分布式算法
      • CAP 定理的含义
  • 分布式
    • 分布式算法
      • Paxos和Raft比较
  • 分布式
    • 分布式算法
      • 分布式一致性与共识算法
  • 分布式
    • 分布式锁
      • ⭐️分布式锁的原理及实现方式
  • 分布式
    • 搜索引擎
      • ElasticSearch与SpringBoot的集成与JPA方法的使用
  • 分布式
    • 搜索引擎
      • 全文搜索引擎 Elasticsearch 入门教程
  • 分布式
    • 搜索引擎
      • 十分钟学会使用 Elasticsearch 优雅搭建自己的搜索系统
  • 分布式
    • 搜索引擎
      • 腾讯万亿级 Elasticsearch 技术解密
  • 分布式
    • 日志系统
      • Grafana Loki 简明教程
  • 分布式
    • 日志系统
      • 分布式系统中如何优雅地追踪日志
  • 分布式
    • 日志系统
      • 如何优雅地记录操作日志?
  • 分布式
    • 日志系统
      • 日志收集组件—Flume、Logstash、Filebeat对比
  • 分布式
    • 日志系统
      • 集中式日志系统 ELK 协议栈详解
  • 分布式
    • 消息队列
      • 消息队列 - Kafka
  • 分布式
    • 消息队列
      • 消息队列 - Kafka、RabbitMQ、RocketMQ等消息中间件的对比
  • 分布式
    • 消息队列
      • 消息队列之 RabbitMQ
  • 分布式
    • 消息队列
      • 消息队列 - 使用docker-compose构建kafka集群
  • 分布式
    • 消息队列
      • 消息队列 - 分布式系统与消息的投递
  • 分布式
    • 消息队列
      • 消息队列 - 如何保证消息的可靠性传输
  • 分布式
    • 消息队列
      • 消息队列 - 如何保证消息的顺序性
  • 分布式
    • 消息队列
      • 消息队列 - 如何保证消息队列的高可用
  • 分布式
    • 消息队列
      • 消息队列 - 消息队列设计精要
  • 分布式
    • 监控系统
      • 深度剖析开源分布式监控CAT
  • 大数据
    • Flink
      • Flink架构与核心组件
  • 微服务
    • Dubbo
      • 基于dubbo的分布式应用中的统一异常处理
  • 微服务
    • Dubbo
      • Vim快捷键
  • 微服务
    • Service Mesh
      • Istio 是什么?
  • 微服务
    • Service Mesh
      • OCTO 2.0:美团基于Service Mesh的服务治理系统详解
  • 微服务
    • Service Mesh
      • Service Mesh是什么?
  • 微服务
    • Service Mesh
      • Spring Cloud向Service Mesh迁移
  • 微服务
    • Service Mesh
      • 数据挖掘算法
  • 微服务
    • Service Mesh
      • Seata Saga 模式
  • 微服务
    • Spring Cloud
      • Seata TCC 模式
  • 微服务
    • Spring Cloud
      • Spring Cloud Config
  • 微服务
    • Spring Cloud
      • Seata AT 模式
  • 微服务
    • Spring Cloud
      • Spring Cloud Gateway
  • 微服务
    • Spring Cloud
      • Spring Cloud OpenFeign 的核心原理
  • 微服务
    • Spring Cloud
      • Seata XA 模式
  • 数据库
    • Database Version Control
      • Liquibase vs. Flyway
  • 数据库
    • Database Version Control
      • Six reasons to version control your database
  • 数据库
    • MySQL
      • How Sharding Works
  • 数据库
    • MySQL
      • MySQL InnoDB中各种SQL语句加锁分析
  • 数据库
    • MySQL
      • MySQL 事务隔离级别和锁
  • 数据库
    • MySQL
      • MySQL 索引性能分析概要
  • 数据库
    • MySQL
      • MySQL 索引设计概要
  • 数据库
    • MySQL
      • MySQL出现Waiting for table metadata lock的原因以及解决方法
  • 数据库
    • MySQL
      • MySQL的Limit性能问题
  • 数据库
    • MySQL
      • MySQL索引优化explain
  • 数据库
    • MySQL
      • MySQL索引背后的数据结构及算法原理
  • 数据库
    • MySQL
      • MySQL行转列、列转行问题
  • 数据库
    • MySQL
      • 一条SQL更新语句是如何执行的?
  • 数据库
    • MySQL
      • 一条SQL查询语句是如何执行的?
  • 数据库
    • MySQL
      • 为什么 MySQL 使用 B+ 树
  • 数据库
    • MySQL
      • 为什么 MySQL 的自增主键不单调也不连续
  • 数据库
    • MySQL
      • 为什么我的MySQL会“抖”一下?
  • 数据库
    • MySQL
      • 为什么数据库不应该使用外键
  • 数据库
    • MySQL
      • 为什么数据库会丢失数据
  • 数据库
    • MySQL
      • 事务的可重复读的能力是怎么实现的?
  • 数据库
    • MySQL
      • 大众点评订单系统分库分表实践
  • 数据库
    • MySQL
      • 如何保证缓存与数据库双写时的数据一致性?
  • 数据库
    • MySQL
      • 浅谈数据库并发控制 - 锁和 MVCC
  • 数据库
    • MySQL
      • 深入浅出MySQL 中事务的实现
  • 数据库
    • MySQL
      • 浅入浅出MySQL 和 InnoDB
  • 数据库
    • PostgreSQL
      • PostgreSQL upsert功能(insert on conflict do)的用法
  • 数据库
    • Redis
      • Redis GEO & 实现原理深度分析
  • 数据库
    • Redis
      • Redis 和 I/O 多路复用
  • 数据库
    • Redis
      • Redis分布式锁
  • 数据库
    • Redis
      • Redis实现分布式锁中的“坑”
  • 数据库
    • Redis
      • Redis总结
  • 数据库
    • Redis
      • 史上最全Redis高可用技术解决方案大全
  • 数据库
    • Redis
      • Redlock:Redis分布式锁最牛逼的实现
  • 数据库
    • Redis
      • 为什么 Redis 选择单线程模型
  • 数据库
    • TiDB
      • 新一代数据库TiDB在美团的实践
  • 数据库
    • 数据仓库
      • 实时数仓在有赞的实践
  • 数据库
    • 数据库原理
      • OLTP与OLAP的关系是什么?
  • 数据库
    • 数据库原理
      • 为什么 OLAP 需要列式存储
  • 系统设计
    • DDD
      • Domain Primitive
  • 系统设计
    • DDD
      • Repository模式
  • 系统设计
    • DDD
      • 应用架构
  • 系统设计
    • DDD
      • 聊聊如何避免写流水账代码
  • 系统设计
    • DDD
      • 领域层设计规范
  • 系统设计
    • DDD
      • 从三明治到六边形
  • 系统设计
    • DDD
      • 阿里盒马领域驱动设计实践
  • 系统设计
    • DDD
      • 领域驱动设计(DDD)编码实践
  • 系统设计
    • DDD
      • 领域驱动设计在互联网业务开发中的实践
  • 系统设计
    • 基础架构
      • 容错,高可用和灾备
  • 系统设计
    • 数据聚合
      • GraphQL及元数据驱动架构在后端BFF中的实践
  • 系统设计
    • 数据聚合
      • 高效研发-闲鱼在数据聚合上的探索与实践
  • 系统设计
    • 服务安全
      • JSON Web Token 入门教程
  • 系统设计
    • 服务安全
      • 你还在用JWT做身份认证嘛?
  • 系统设计
    • 服务安全
      • 凭证(Credentials)
  • 系统设计
    • 服务安全
      • 授权(Authorization)
  • 系统设计
    • 服务安全
      • 理解OAuth2.0
  • 系统设计
    • 服务安全
      • 认证(Authentication)
  • 系统设计
    • 架构案例
      • 微信 Android 客户端架构演进之路
  • 系统设计
    • 高可用架构
      • 业务高可用的保障:异地多活架构
  • 计算机基础
    • 字符编码
      • Base64原理解析
  • 计算机基础
    • 字符编码
      • 字符编码笔记:ASCII,Unicode 和 UTF-8
  • 计算机基础
    • 操作系统
      • 为什么 CPU 访问硬盘很慢
  • 计算机基础
    • 操作系统
      • 为什么 HTTPS 需要 7 次握手以及 9 倍时延
  • 计算机基础
    • 操作系统
      • 为什么 Linux 默认页大小是 4KB
  • 计算机基础
    • 操作系统
      • 磁盘IO那些事
  • 计算机基础
    • 操作系统
      • 虚拟机的3种网络模式
  • 计算机基础
    • 服务器
      • mac终端bash、zsh、oh-my-zsh最实用教程
  • 计算机基础
    • 服务器
      • Nginx强制跳转Https
  • 计算机基础
    • 服务器
      • curl 的用法指南
  • 计算机基础
    • 网络安全
      • 如何设计一个安全的对外接口?
  • 计算机基础
    • 网络安全
      • 浅谈常见的七种加密算法及实现
  • 计算机基础
    • 网络编程
      • MQTT - The Standard for IoT Messaging
  • 计算机基础
    • 网络编程
      • 两万字长文 50+ 张趣图带你领悟网络编程的内功心法
  • 计算机基础
    • 网络编程
      • 为什么 TCP 协议有 TIME_WAIT 状态
  • 计算机基础
    • 网络编程
      • 为什么 TCP 协议有性能问题
  • 计算机基础
    • 网络编程
      • 为什么 TCP 协议有粘包问题
  • 计算机基础
    • 网络编程
      • 为什么 TCP 建立连接需要三次握手
  • 计算机基础
    • 网络编程
      • 为什么 TCP/IP 协议会拆分数据
  • 计算机基础
    • 网络编程
      • 使用 OAuth 2 和 JWT 为微服务提供安全保障
  • 计算机基础
    • 网络编程
      • 四种常见的 POST 提交数据方式
  • 计算机基础
    • 网络编程
      • 有赞TCP网络编程最佳实践
  • 计算机基础
    • 网络编程
      • 看完这篇HTTP,跟面试官扯皮就没问题了
  • 计算机基础
    • 网络编程
      • 详细解析 HTTP 与 HTTPS 的区别
  • 质量&效率
    • 快捷键
      • Idea快捷键(Mac版)
  • 质量&效率
    • 快捷键
      • Shell快捷键
  • 质量&效率
    • 快捷键
      • conduit
  • 质量&效率
    • 敏捷开发
      • Scrum的3种角色
  • 质量&效率
    • 敏捷开发
      • Scrum的4种会议
  • 质量&效率
    • 敏捷开发
      • ThoughtWorks的敏捷开发
  • 质量&效率
    • 敏捷开发
      • 敏捷开发入门教程
  • 运维&测试
    • Docker
      • Docker (容器) 的原理
  • 运维&测试
    • Docker
      • Docker Compose:链接外部容器的几种方式
  • 运维&测试
    • Docker
      • Docker 入门教程
  • 运维&测试
    • Docker
      • Docker 核心技术与实现原理
  • 运维&测试
    • Docker
      • Dockerfile 最佳实践
  • 运维&测试
    • Docker
      • Docker开启Remote API 访问 2375端口
  • 运维&测试
    • Docker
      • Watchtower - 自动更新 Docker 镜像与容器
  • 运维&测试
    • Kubernetes
      • Kubernetes 介绍
  • 运维&测试
    • Kubernetes
      • Kubernetes 在有赞的实践
  • 运维&测试
    • Kubernetes
      • Kubernetes 学习路径
  • 运维&测试
    • Kubernetes
      • Kubernetes如何改变美团的云基础设施?
  • 运维&测试
    • Kubernetes
      • Kubernetes的三种外部访问方式:NodePort、LoadBalancer 和 Ingress
  • 运维&测试
    • Kubernetes
      • 谈 Kubernetes 的架构设计与实现原理
  • 运维&测试
    • 压测
      • 全链路压测平台(Quake)在美团中的实践
  • 运维&测试
    • 测试
      • Cpress - JavaScript End to End Testing Framework
  • 运维&测试
    • 测试
      • 代码覆盖率-JaCoCo
  • 运维&测试
    • 测试
      • 浅谈代码覆盖率
  • 运维&测试
    • 测试
      • 测试中 Fakes、Mocks 以及 Stubs 概念明晰
  • Java FrameWorks
    • Spring
      • Spring AOP
        • Spring AOP中的Bean是如何被AOP代理的
  • Java FrameWorks
    • Spring
      • Spring AOP
        • Spring AOP原生动态代理和Cglib动态代理
  • Java FrameWorks
    • Spring
      • Spring AOP
        • Spring AOP实现方式(xml&注解)
  • Java FrameWorks
    • Spring
      • Spring AOP
        • Spring AOP是如何收集切面类并封装的
  • Java FrameWorks
    • Spring
      • Spring AOP
        • Spring AOP概述
  • Java FrameWorks
    • Spring
      • Spring AOP
        • Spring AOP的底层核心后置处理器
  • Java FrameWorks
    • Spring
      • Spring AOP
        • Spring AOP的延伸知识
  • Java FrameWorks
    • Spring
      • Spring Boot
        • Spring Boot - IOC(一)
  • Java FrameWorks
    • Spring
      • Spring Boot
        • Spring Boot - IOC(三)
  • Java FrameWorks
    • Spring
      • Spring Boot
        • Spring Boot - IOC(二)
  • Java FrameWorks
    • Spring
      • Spring Boot
        • Spring Boot - IOC(五)
  • Java FrameWorks
    • Spring
      • Spring Boot
        • Spring Boot - IOC(四) - 循环依赖与解决方案
  • Java FrameWorks
    • Spring
      • Spring Boot
        • Spring Boot - 启动引导
  • Java FrameWorks
    • Spring
      • Spring Boot
        • Spring Boot JarLauncher
  • Java FrameWorks
    • Spring
      • Spring Boot
        • Spring Boot Web Mvc 自动装配
  • Java FrameWorks
    • Spring
      • Spring Boot
        • Spring Boot 使用ApplicationListener监听器
  • Java FrameWorks
    • Spring
      • Spring Boot
        • Spring Boot 声明式事务
  • Java FrameWorks
    • Spring
      • Spring Boot
        • Spring Boot 嵌入式容器
  • Java FrameWorks
    • Spring
      • Spring Boot
        • Spring Boot引起的“堆外内存泄漏”排查及经验总结
  • Java FrameWorks
    • Spring
      • Spring Boot
        • Spring Boot的启动流程
  • Java FrameWorks
    • Spring
      • Spring Boot
        • Spring Boot自动化配置源码分析
  • Java FrameWorks
    • Spring
      • Spring Boot
        • 如何自定义Spring Boot Starter?
  • Java FrameWorks
    • Spring
      • Spring IOC
        • IOC - 模块装配和条件装配
  • Java FrameWorks
    • Spring
      • Spring IOC
        • IOC - 配置源(xml,注解)
  • Java FrameWorks
    • Spring
      • Spring IOC
        • Spring Environment
  • Java FrameWorks
    • Spring
      • Spring IOC
        • Spring ApplicationContext
  • Java FrameWorks
    • Spring
      • Spring IOC
        • Spring BeanDefinition
  • Java FrameWorks
    • Spring
      • Spring IOC
        • Spring BeanFactory
  • Java FrameWorks
    • Spring
      • Spring IOC
        • Spring BeanFactoryPostProcessor
  • Java FrameWorks
    • Spring
      • Spring IOC
        • Spring BeanPostProcessor
  • Java FrameWorks
    • Spring
      • Spring IOC
        • Spring Bean的生命周期(一) - 概述
  • Java FrameWorks
    • Spring
      • Spring IOC
        • Spring Bean的生命周期(三) - 实例化阶段
  • Java FrameWorks
    • Spring
      • Spring IOC
        • Spring Bean的生命周期(二) - BeanDefinition
  • Java FrameWorks
    • Spring
      • Spring IOC
        • Spring Bean的生命周期(五) - 销毁阶段
  • Java FrameWorks
    • Spring
      • Spring IOC
        • Spring Bean的生命周期(四) - 初始化阶段
  • Java FrameWorks
    • Spring
      • Spring IOC
        • Spring ComponentScan
  • Java FrameWorks
    • Spring
      • Spring IOC
        • Spring Events
  • Java FrameWorks
    • Spring
      • Spring IOC
        • Spring IOC 基础篇
  • Java FrameWorks
    • Spring
      • Spring IOC
        • Spring IOC 总结
  • Java FrameWorks
    • Spring
      • Spring IOC
        • Spring IOC 进阶篇
  • Java FrameWorks
    • Spring
      • Spring IOC
        • Spring IOC容器的生命周期
  • Java FrameWorks
    • Spring
      • Spring IOC
        • Spring Resource
  • Java FrameWorks
    • Spring
      • Spring MVC
        • DispatcherServlet的初始化原理
  • Java FrameWorks
    • Spring
      • Spring MVC
        • DispatcherServlet的核心工作原理
  • Java FrameWorks
    • Spring
      • Spring MVC
        • WebMvc的架构设计与组件功能解析
  • Java FrameWorks
    • Spring
      • Spring Security
        • Spring Boot 2 + Spring Security 5 + JWT 的单页应用 Restful 解决方案
  • Java FrameWorks
    • Spring
      • Spring Security
        • Spring Security Oauth
  • Java FrameWorks
    • Spring
      • Spring Security
        • Spring Security
  • Java FrameWorks
    • Spring
      • Spring WebFlux
        • DispatcherHandler的工作原理(传统方式)
  • Java FrameWorks
    • Spring
      • Spring WebFlux
        • DispatcherHandler的工作原理(函数式端点)
  • Java FrameWorks
    • Spring
      • Spring WebFlux
        • WebFlux的自动装配
  • Java FrameWorks
    • Spring
      • Spring WebFlux
        • 快速了解响应式编程与Reactive
  • Java FrameWorks
    • Spring
      • Spring WebFlux
        • 快速使用WebFlux
  • 分布式
    • 协调服务
      • Zookeeper
        • Zookeeper - 客户端之 Curator
  • 分布式
    • 协调服务
      • Zookeeper
        • 详解分布式协调服务 ZooKeeper
  • 分布式
    • 协调服务
      • etcd
        • 高可用分布式存储 etcd 的实现原理
  • 数据库
    • Database Version Control
      • Flyway
        • Database Migrations with Flyway
  • 数据库
    • Database Version Control
      • Flyway
        • How Flyway works
  • 数据库
    • Database Version Control
      • Flyway
        • Rolling Back Migrations with Flyway
  • 数据库
    • Database Version Control
      • Flyway
        • The meaning of the concept of checksums
  • 数据库
    • Database Version Control
      • Liquibase
        • Introduction to Liquibase Rollback
  • 数据库
    • Database Version Control
      • Liquibase
        • LiquiBase中文学习指南
  • 数据库
    • Database Version Control
      • Liquibase
        • Use Liquibase to Safely Evolve Your Database Schema
  • 系统设计
    • 流量控制
      • RateLimiter
        • Guava Rate Limiter实现分析
  • 系统设计
    • 流量控制
      • Sentinel
        • Sentinel 与 Hystrix 的对比
  • 系统设计
    • 流量控制
      • Sentinel
        • Sentinel工作主流程
  • 系统设计
    • 流量控制
      • 算法
        • 分布式服务限流实战
  • 系统设计
    • 解决方案
      • 秒杀系统
        • 如何设计一个秒杀系统
  • 系统设计
    • 解决方案
      • 红包系统
        • 微信高并发资金交易系统设计方案--百亿红包背后的技术支撑
  • 计算机基础
    • 数据结构与算法
      • 其他相关
        • 什么是预排序遍历树算法(MPTT,Modified Preorder Tree Traversal)
  • 计算机基础
    • 数据结构与算法
      • 其他相关
        • 加密算法
  • 计算机基础
    • 数据结构与算法
      • 其他相关
        • 推荐系统算法
  • 计算机基础
    • 数据结构与算法
      • 其他相关
        • linkerd
  • 计算机基础
    • 数据结构与算法
      • 其他相关
        • 查找算法
  • 计算机基础
    • 数据结构与算法
      • 其他相关
        • 缓存淘汰算法中的LRU和LFU
  • 计算机基础
    • 数据结构与算法
      • 其他相关
        • 负载均衡算法
  • 计算机基础
    • 数据结构与算法
      • 分布式算法
        • 分布式算法 - Paxos算法
  • 计算机基础
    • 数据结构与算法
      • 分布式算法
        • 分布式算法 - Raft算法
  • 计算机基础
    • 数据结构与算法
      • 分布式算法
        • 分布式算法 - Snowflake算法
  • 计算机基础
    • 数据结构与算法
      • 分布式算法
        • 分布式算法 - ZAB算法
  • 计算机基础
    • 数据结构与算法
      • 分布式算法
        • 分布式算法 - 一致性Hash算法
  • 计算机基础
    • 数据结构与算法
      • 大数据处理
        • 大数据处理 - Bitmap & Bloom Filter
  • 计算机基础
    • 数据结构与算法
      • 大数据处理
        • 大数据处理 - Map & Reduce
  • 计算机基础
    • 数据结构与算法
      • 大数据处理
        • 大数据处理 - Trie树/数据库/倒排索引
  • 计算机基础
    • 数据结构与算法
      • 大数据处理
        • 大数据处理 - 分治/hash/排序
  • 计算机基础
    • 数据结构与算法
      • 大数据处理
        • 大数据处理 - 双层桶划分
  • 计算机基础
    • 数据结构与算法
      • 大数据处理
        • 大数据处理 - 外(磁盘文件)排序
  • 计算机基础
    • 数据结构与算法
      • 大数据处理
        • 大数据处理 - 布隆过滤器
  • 计算机基础
    • 数据结构与算法
      • 大数据处理
        • 大数据处理算法
  • 计算机基础
    • 数据结构与算法
      • 字符串匹配算法
        • 字符串匹配 - 文本预处理:后缀树(Suffix Tree)
  • 计算机基础
    • 数据结构与算法
      • 字符串匹配算法
        • 字符串匹配 - 模式预处理:BM 算法 (Boyer-Moore)
  • 计算机基础
    • 数据结构与算法
      • 字符串匹配算法
        • 字符串匹配 - 模式预处理:KMP 算法(Knuth-Morris-Pratt)
  • 计算机基础
    • 数据结构与算法
      • 字符串匹配算法
        • 字符串匹配 - 模式预处理:朴素算法(Naive)(暴力破解)
  • 计算机基础
    • 数据结构与算法
      • 字符串匹配算法
        • 字符串匹配
  • 计算机基础
    • 数据结构与算法
      • 常用算法
        • 分支限界算法
  • 计算机基础
    • 数据结构与算法
      • 常用算法
        • 分治算法
  • 计算机基础
    • 数据结构与算法
      • 常用算法
        • 动态规划算法
  • 计算机基础
    • 数据结构与算法
      • 常用算法
        • 回溯算法
  • 计算机基础
    • 数据结构与算法
      • 常用算法
        • 贪心算法
  • 计算机基础
    • 数据结构与算法
      • 排序算法
        • 十大排序算法
  • 计算机基础
    • 数据结构与算法
      • 排序算法
        • 图解排序算法(一)之3种简单排序(选择,冒泡,直接插入)
  • 计算机基础
    • 数据结构与算法
      • 排序算法
        • 图解排序算法(三)之堆排序
  • 计算机基础
    • 数据结构与算法
      • 排序算法
        • 图解排序算法(二)之希尔排序
  • 计算机基础
    • 数据结构与算法
      • 排序算法
        • 图解排序算法(四)之归并排序
  • 计算机基础
    • 数据结构与算法
      • 数据结构
        • 树的高度和深度
  • 计算机基础
    • 数据结构与算法
      • 数据结构
        • 红黑树深入剖析及Java实现
  • 计算机基础
    • 数据结构与算法
      • 数据结构
        • 线性结构 - Hash
  • 计算机基础
    • 数据结构与算法
      • 数据结构
        • 线性结构 - 数组、链表、栈、队列
  • 计算机基础
    • 数据结构与算法
      • 数据结构
        • 逻辑结构 - 树
  • 运维&测试
    • 测试
      • Spock
        • Groovy 简明教程
  • 运维&测试
    • 测试
      • Spock
        • Spock 官方文档
  • 运维&测试
    • 测试
      • Spock
        • Spock单元测试框架介绍以及在美团优选的实践
  • 运维&测试
    • 测试
      • TDD
        • TDD 实践 - FizzFuzzWhizz(一)
  • 运维&测试
    • 测试
      • TDD
        • TDD 实践 - FizzFuzzWhizz(三)
  • 运维&测试
    • 测试
      • TDD
        • TDD 实践 - FizzFuzzWhizz(二)
  • 运维&测试
    • 测试
      • TDD
        • 测试驱动开发(TDD)- 原理篇
  • 微服务
    • Spring Cloud
      • Spring Cloud Alibaba
        • Nacos
          • Nacos 服务注册的原理
  • 微服务
    • Spring Cloud
      • Spring Cloud Alibaba
        • Nacos
          • Nacos 配置中心原理分析
  • 微服务
    • Spring Cloud
      • Spring Cloud Alibaba
        • Seata
          • 服务调用过程
  • 微服务
    • Spring Cloud
      • Spring Cloud Alibaba
        • Seata
          • Spring Cloud Bus
  • 微服务
    • Spring Cloud
      • Spring Cloud Alibaba
        • Seata
          • Spring Cloud Consul
  • 微服务
    • Spring Cloud
      • Spring Cloud Alibaba
        • Seata
          • Spring Cloud Stream
  • 微服务
    • Spring Cloud
      • Spring Cloud Alibaba
        • Sentinel
          • Sentinel 与 Hystrix 的对比
  • 微服务
    • Spring Cloud
      • Spring Cloud Alibaba
        • Sentinel
          • Sentinel
  • 微服务
    • Spring Cloud
      • Spring Cloud Netflix
        • Hystrix
          • How Hystrix Works
  • 微服务
    • Spring Cloud
      • Spring Cloud Netflix
        • Hystrix
          • Hystrix
  • 微服务
    • Spring Cloud
      • Spring Cloud Netflix
        • Hystrix
          • Hystrix原理与实战
  • 微服务
    • Spring Cloud
      • Spring Cloud Netflix
        • Hystrix
          • Spring Cloud Hystrix基本原理
由 GitBook 提供支持
在本页
  • 1. 说明
  • 2. Groovy 语言基础
  • 2.1 Groovy 和 Java 的一些区别
  • 3. Spock
  • 3.1 maven
  • 3.2 导入
  • 3.3 声明测试类
  • 3.4 字段
  • 3.5 固定方法
  • 3.6 测试方法
  • 3.7 辅助方法
  • 3.8 with() 方法
  • 3.9 文档化
  • 3.10 基础扩展注解
  • 3.11 和 JUnit 的比较
  • 4. 数据驱动测试
  • 4.1 概要
  • 4.2 数据表
  • 4.3 测试用例的隔离性
  • 4.4 用例之间的对象共享
  • 4.5 语法改进
  • 4.6 @Unroll
  • 4.7 数据管道
  • 4.8 多值数据管道
  • 4.9 测试变量赋值
  • 4.10 数据表,数据管道和变量赋值混用
  • 5. 交互驱动测试
  • 5.1 创建 Mock 对象
  • 5.2 Mock 对象的默认行为
  • 5.3 向被测对象注入 Mock 对象
  • 5.4 Mocking
  • 5.5 Stubbing
  • 5.6 结合 Mocking 和 Stubbing
  • 5.7 其他
  • 6. 进阶扩展注解
  • 6.1 IgnoreIf
  • 6.2 Requires
  • 6.3 Retry
  • 7. 与spring-boot集成
  • 7.1 maven
  • 7.2 测试spring-boot接口

这有帮助吗?

  1. 运维&测试
  2. 测试
  3. Spock

Spock 官方文档

上一页Spock下一页运维&测试

最后更新于2年前

这有帮助吗?

转载:

1. 说明

本文档为 Spock 官方文档的大致翻译,很多地方没有直接对应,官方文档参考:

2. Groovy 语言基础

Spock 是基于 Groovy 的测试框架。Groovy 是运行在 JVM 上的脚本语言,语法和 Java 兼容,可以理解为脚本化的 Java。Groovy 教程参考: https://www.w3cschool.cn/groovy

2.1 Groovy 和 Java 的一些区别

  1. public 修饰符在 Groovy 中不是必须的。

  2. Groovy 支持范围类型,比如 1..3 相当于 Java 中的数组 { 1, 2, 3 };1.2..3 相当于 Java 中的数组 { 1.2, 2.2 };1..<3 相当于数组 { 1, 2 };'a'..'z' 相当于所有小写字母的字符集合。范围类型是可迭代的。

  3. Groovy 中的数组(列表)表示为 [ v1, v2 ],用中括号括起。

  4. Groovy 中的字典可表示为 [ k1: v1, k2: v2 ],键和值的类型都可以随意,可以迭代键值对。空字典 [: ]。

  5. Groovy 支持类型自动推断,使用关键字 def,比如 def var = value;。

  6. Groovy 的方法的返回值可以指定为某一类型,也可以指定为 def。

  7. Groovy 的方法支持默认参数。

  8. 一个 Groovy 脚本就是一个类,定义在脚本中的类可以理解为内部类,直接定义在脚本中的方法(或者说是函数)可以当成脚本类的成员方法。

  9. 字符串中 $xxx 可以直接引用变量。

  10. 字符串中 ${xxx.xxx()} 可以直接引用方法的返回值。

  11. Groovy 支持范围下标,负数下标。

  12. Groovy 可以使用 Java I/O 的所有类,除此之外还有 I/O 操作的快捷方式:

    • 对文件进行逐行处理:

      // java.io.File
      new File(filepath).eachLine {
          line -> processLine(line)
      };
    • 获取文件的整个文本:

      // java.io.File
      new File(filepath).text
    • 写入文件:

      new File(filepath).withWriter("UTF-8") {
          writer -> writer.write(str)
      };
    • 获取文件大小(单位:字节):

      new File(filepath).length()
    • 判断路径是文件还是目录:

      file.isDirectory()
      file.isFile()
    • 文件内容复制:

      fileCopy << file.text // 不能 file.text >> fileCopy
  13. def regex = ~"reg" 创建正则表达式对象。strValue ==~ "regPattern" 判断字符串整体是否匹配右侧的正则表达式,返回布尔值。strValue =~ "regPattern" 创建 Matcher 对象。

  14. Groovy 中的除法默认是小数除法,不是整数除法。

  15. Groovy 中的闭包可以表示为

    {
      函数体
    }

    或

    {
      (param1, param2, ... ) -> 函数体
    }

    闭包是一种对象,通过 closure.call(..) 调用闭包,闭包中可以调用闭包外的变量。所有闭包都有一个名为 it 的隐式参数。

  16. Groovy 中支持类似于 python 中的字典传参,比如 func(k1: v1, k2: v2, ...),传入的参数就是一个字典 [k1: v1, k2: v2, ...]。

  17. Groovy 中形参可以不用指定类型。

  18. Java 中通过 .class 获取类对象,在 Groovy 中可以省略。

  19. Groovy 支持 ?. 语法,比如 foo?.bar(),当 foo 为 null 时,整个表达式返回 null,否则就调用 bar()。

  20. Groovy 中 return 可以省略。

3. Spock

3.1 maven

<dependency>
    <groupId>org.spockframework</groupId>
    <artifactId>spock-core</artifactId>
    <version>1.1-groovy-2.4</version>
    <scope>test</scope>
</dependency>

3.2 导入

导入 Spock 框架基础包 import spock.lang.*。

3.3 声明测试类

class MyFirstSpec extends Specification {
  // 字段
  // 固定方法
  // 测试方法
  // 辅助方法
}

测试类是一个 Groovy 类,必须继承 spock.lang.Specification,规范的测试类名称一般是XxxSpec。

3.4 字段

def obj = new ClassUnderSpecification()
def coll = new Collaborator()

非静态字段可以在固定方法中使用,推荐在声明时同时初始化(在 setup() 方法中初始化也是一样的)。但是不能在测试方法之间共享,每个测试方法都有一个隔离的副本,每个测试方法开始前都会初始化一遍。

@Shared res = new VeryExpensiveResource()

但是有时候又需要在测试方法之间共享字段,比如这个字段的初始化的开销很大,或者需要在测试方法间通信。通过将字段标记为 @Shared,可以使字段成为共享字段。还是推荐在声明时初始化(在 setupSpec() 方法中初始化也是一样的)。

static final PI = 3.141592654

静态字段只应该用作常量。共享字段完全可以取代静态变量的作用,且共享字段在语义上更明确。

3.5 固定方法

def setup() {}          // 每个测试方法开始前都会执行一遍
def cleanup() {}        // 每个测试方法后都会执行一遍
def setupSpec() {}     // 在第一个测试方法开始前执行一遍
def cleanupSpec() {}   // 最后一个测试方法后执行

固定方法负责测试方法和测试类的环境初始化和资源清理工作。固定方法可以有也可以没有,但是建议要有 setup() 和 cleanup()。注意 setupSpec() 和 cleanupSpec() 中不可以引用非共享字段。

这四个固定方法其实是重写自 spock.lang.Specification,setup() 和 setupSpec() 的执行顺序是先父类后子类,cleanup() 和 cleanupSpec() 的执行顺序是先子类后父类。不需要显式调用父类的对应方法。

3.6 测试方法

def "pushing an element on the stack"() {
  // 测试方法
}

测试方法是测试类的核心。测试方法名是一个字符串常量,也是对测试方法的描述。一个测试方法应该由4部分组成:

  1. Setup:环境初始化(可选)

  2. Stimulus:调用待测试代码

  3. Response:描述预期行为

  4. Cleanup:清理资源(可选)

Spock 对这四个概念上的阶段提供了内建的支持,也就是“块”(block)。一个块从一个标签开始直至另一个标签或方法结尾。有6个块:setup,when,then,expect,cleanup,where。从方法的开头到第一个标签之间的所有语句都属于隐式的 setup 块。

一个测试方法至少要有一个显式声明的块,一个测试方法之所以是一个测试方法就是因为它有显式块。块把测试方法划分成不同的区域,且块不可以嵌套。

下图展示了具体的块和测试方法的4个概念阶段的对应关系:

其中,where 块比较特殊,后面会再提,先看其他5个块。

3.6.1 setup 块

setup:
def stack = new Stack()
def elem = "push me"

在 setup 块中应对当前测试方法的环境进行初始化。该块是一定是第一个执行的,且不会由于其他测试方法的存在而反复执行(和 setup() 的区别)。setup: 标签可以省略。given: 标签是 setup: 标签的别名。

3.6.2 when 和 then 块

when:   // 调用待测代码
then:   // 描述期望的行为

when 和 then 块是绑定在一块使用的,这两个块组合在一块使用可以调用待测代码并指定期望的行为。when 块可以是任意代码,但是 then 中的语句仅限于布尔表达式(省略了 assert),异常情况判断语句(thrown() 和 notThrown())和变量定义语句。一个测试方法中可以有多个 when-then 块。注意 then 块不宜过于庞大,尽量在5个语句之内。

关于 thrown() 和 notThrown()

这两个方法用于描述 when 块中是否应该抛出异常,参数传入异常类型。比如从一个空栈中弹栈应该抛出 EmptyStackException,为了描述这一行为,应该这样写:

when:
stack.pop()

then:
thrown(EmptyStackException)
stack.empty

thrown() 后可以跟其他条件和其他块。特别地,thrown() 返回抛出的异常:

when:
stack.pop()

then:
def e = thrown(EmptyStackException)
e.cause == null

或者另一种写法:

when:
stack.pop()

then:
EmptyStackException e = thrown()
e.cause == null

推荐第二种写法,因为可读性更强,且变量 e 的类型明确指定,方便 IDE 提供代码补全。

有时候我们期望的行为是没有任何异常出现,比如 HashMap 可以接受一个 null 键:

def "HashMap accepts null key"() {
  setup:
  def map = new HashMap()
  map.put(null, "elem")
}

这样写是可以的但是不好,因为没有明确地指出期望的行为,即没有异常出现。这样写更好:

def "HashMap accepts null key"() {
  setup:
  def map = new HashMap()

  when:
  map.put(null, "elem")

  then:
  notThrown(NullPointerException)
}

通过 notThrown() 可以更明确地指出 when 块不期望抛出 NullPointerException 异常。如果抛出了其他类型的异常,测试依然会不通过。

3.6.3 expect 块

expect 块可以看成是 then 块的一种简化版,只能包含布尔表达式和变量定义。当调用待测代码和描述期望行为都很简单,在一个表达式中就能搞定时,用 expect 块更简单。举个栗子,测试 Math.max() 方法:

when:
def x = Math.max(1, 2)

then:
x == 2
expect:
Math.max(1, 2) == 2

两种方式都是一样的,但是明显第二种更简洁。

3.6.4 cleanup 块

setup:
def file = new File("/some/path")
file.createNewFile()

// ...

cleanup:
file.delete()

cleanup 块后面只能跟 where 块,或放在测试方法最后,且不会因为其他测试方法的存在而被反复调用(和 cleanup() 的区别),用于清理当前测试方法使用的资源。即使测试没有通过,该块也会执行(类似 Java 中的 finally)。cleanup 块用到较少,一般是在关闭流,关闭数据库连接或关闭网络服务的时候才会用到。

如果所有测试方法清理资源的逻辑都是一样的,就用 cleanup(),否则就用 cleanup 块。setup() 和 setup 块也是同理。

3.6.5 where 块

where 块一定是放在方法的最后,如果有的话,用于写数据驱动的测试方法。举个栗子:

def "computing the maximum of two numbers"() {
  expect:
  Math.max(a, b) == c

  where:
  a << [5, 3]
  b << [1, 9]
  c << [5, 9]
}

where 块为当前的测试方法创建了两个测试用例,很简洁,一个 a 是 5,b 是 1,c 是 5;另一个 a 是 3,b 是 9,c 是 9。

虽然 where 块放在最后,但却是第一个执行的。where 块在数据驱动测试一章会有更详细的说明。

3.7 辅助方法

一些情况下,测试方法很庞大或包含大量冗余代码,需要辅助方法抽离代码,比如:

def "offered PC matches preferred configuration"() {
  when:
  def pc = shop.buyPc()

  then:
  pc.vendor == "Sunny"
  pc.clockRate >= 2333
  pc.ram >= 4096
  pc.os == "Linux"
}

将大量条件抽离:

def "offered PC matches preferred configuration"() {
  when:
  def pc = shop.buyPc()

  then:
  matchesPreferredConfiguration(pc)
}

def matchesPreferredConfiguration(pc) {
  pc.vendor == "Sunny"
  && pc.clockRate >= 2333
  && pc.ram >= 4096
  && pc.os == "Linux"
}

辅助方法 matchesPreferredConfiguration() 中只有一个布尔表达式,且将该表达式的值返回(省略了 return)。这种写法有个严重的问题,如果有一个条件不满足,没法知道具体是哪一个。改进:

void matchesPreferredConfiguration(pc) {
  assert pc.vendor == "Sunny"
  assert pc.clockRate >= 2333
  assert pc.ram >= 4096
  assert pc.os == "Linux"
}

注意两点:

  1. assert 不能省略。

  2. 返回类型必须是 void。

一些建议:代码重用是好事,但不要过度。使用固定方法和辅助方法会增加测试方法之间的耦合度。如果代码重用过多,你会发现测试方法难以维护。

3.8 with() 方法

作为辅助方法的替代方案,可以调用 with(target, closure) 来验证对象,仅在 expect 或 then 块中才能调用:

def "offered PC matches preferred configuration"() {
  when:
  def pc = shop.buyPc()

  then:
  with(pc) {
    vendor == "Sunny"
    clockRate >= 2333
    ram >= 406
    os == "Linux"
  }
}

不需要像辅助方法中那样写很多 assert 了。

3.9 文档化

一份编写规范的测试代码应该具有很强的可读性,尤其是当这些测试代码需要被开发者之外的很多人阅读的时候,比如产品经理,架构师,客户等。Spock 不仅可以通过方法名描述测试方法,还能描述测试方法中的各个块:

setup: "open a database connection"
// code goes here

使用 and: 标签分割一个块,并对各个部分分别描述:

setup: "open a database connection"
// code goes here

and: "seed the customer table"
// code goes here

and: "seed the product table"
// code goes here

and: 标签可以插入到测试方法的任意位置,不影响语义。可以理解为它的作用就是描述一段代码,类似注释。

块描述不仅在源码中呈现,在运行时也能呈现(和注释的区别),可作为调试信息。

3.10 基础扩展注解

Spock 还提供了一套扩展的注解:

  • @Timeout:标记测试方法,设置其执行的超时时间,超时就测试不通过。默认单位秒,可以通过 unit 参数指定时间单位。也可以标记测试类,相当于标记了所有测试方法。测试方法上的 @Timeout 会覆盖测试类上的 @Timeout。

  • @Ignore:忽略标记的测试方法。

  • @IgnoreRest:忽略其他没有该注解的测试方法。需要快速运行其中一个测试方法而不需要运行整个测试类时,这注解就很有用。

  • @FailsWith:标记测试方法,表明该测试方法的预期行为就是测试不通过。这个注解一般会在这种情况下用到,就是待测代码存在已知的 bug,且短时间内无法修复。其他情况下用 thrown() 和 notThrown() 更合适。

后面还会提到进阶扩展注解,以及如何实现自定义的扩展注解。

3.11 和 JUnit 的比较

虽然 Spock 使用了不同的技术,但很多概念是受到 JUnit 启发,下面两个框架的比较:

Spock
JUnit

Specification

Test class

setup()

@Before

cleanup()

@After

setupSpec()

@BeforeClass

cleanupSpec()

@AfterClass

Feature

Test

Feature method

Test method

Data-driven feature

Theory

Condition

Assertion

Exception condition

@Test(expected=…)

4. 数据驱动测试

很多情况下,需要用一组不同的输入和输出来测试同一份代码。Spock 对数据驱动测试提供了大量支持。

4.1 概要

假设现在要测试 Math.max() 方法:

class MathSpec extends Specification {
    def "maximum of two numbers"() {
        expect:
        // 使用不同的输入输出测试同一个方法
        Math.max(1, 3) == 3
        Math.max(7, 4) == 7
        Math.max(0, 0) == 0
    }
}

这种简单的直接的写法在这种简单的情况下是可以的,但是会有一些潜在的问题。下面重构成数据驱动的测试方法。首先引入三个方法形参(称为数据变量),替换硬编码的输入输出:

class MathSpec extends Specification {
    def "maximum of two numbers"(int a, int b, int c) {
        expect:
        Math.max(a, b) == c

        ...
    }
}

测试逻辑已经写好了,但是测试用例还没有。最普遍的做法是,在 where 块中写一个数据表(跟 Markdown 中的表格很像)。

4.2 数据表

数据表可以很方便地引入一系列的测试用例:

class Math extends Specification {
    def "maximum of two numbers"(int a, int b, int c) {
        expect:
        Math.max(a, b) == c

        where:
        a | b | c
        1 | 3 | 3
        7 | 4 | 7
        0 | 0 | 0
    }
}

表格的第一行是表头,定义了有哪些数据变量。下面的行,一行就是一个测试用例。针对每一行,测试方法都会执行一遍(用例之间是隔离的)。如果一个用例失败,剩余的用例依然会执行,所有的失败都会报告。

数据表至少要有两列,如果只有一个数据变量,可以这么写:

where:
a | _
1 | _
7 | _
0 | _

4.3 测试用例的隔离性

测试用例之间是隔离的,可以理解为有多少用例就有多少个测试方法,只是这些测试方法的逻辑都是一样的,且每个测试用例都会执行 setup() 和 cleanup() 方法。

4.4 用例之间的对象共享

只能通过共享字段和静态字段在用例间共享对象。注意只有共享字段和静态字段才能在 where 块中访问到。

注意到共享字段和静态字段也能共享给其他测试方法。目前还没有好的办法让对象只能在同一测试方法的不同用例之间共享。如果真要解决这个问题,可以考虑将测试方法单独放到一个测试类中(一个测试类中有且仅有一个测试方法),然后将若干个这样的测试类放到一个脚本里。

4.5 语法改进

之前的数据驱动测试方法和数据表可以在语法上稍加改进。首先,where 块中已经定义了数据变量,方法形参可以省略了。其次,输入和输出可以使用双管道符分隔,使看起来更清楚。最后变成这样:

class DataDriven extends Specification {
    def "maximum of two numbers"() {
        expect:
        Math.max(a, b) == c

        where:
        a | b || c
        3 | 5 || 5
        7 | 0 || 7
        0 | 0 || 0
    }
}

4.6 @Unroll

@Unroll 注解用于标记测试方法,作用是报告测试结果时以测试用例为基本单位,而不是测试方法:

@Unroll
def "maximum of two numbers"() { ... }

注意 @Unroll 不会对测试方法的执行产生任何影响,只影响报告测试结果的行为。

进一步地:

@Unroll
def "maximum of #a and #b is #c"() { ... }

以井号(#)开头的测试变量引用,配合 @Unroll,报告测试用例的测试结果时,测试描述中的变量引用,会被替换为相应测试变量的具体值。注意必须要两者配合才有用,没有 @Unroll 的话,#xxx 会原样显示。

# 后还可以是对象属性访问和无参方法调用,比如有一个测试变量 person 是 Person 类型的对象,其中有两个属性 name 和 age,则下面的写法是正确的:

def "#person is #person.age years old"() { ... } // 对象属性访问
def "#person.name.toUpperCase()"() { ... } // 无参方法调用

下面的写法的错误的:

def "#person.name.split(' ')[1]" { ... } // 方法有参数,不行
def "#person.age / 2" { ... } // 四则运算不行

但是可以通过额外的测试变量,在描述中引用复杂表达式的结果:

def "#lastName"() {
    ...
    where:
    person << ...
    lastName = person.name.split(' ')[1]
}

@Unroll 也可以标记测试类,相当于标记了该测试类中的每个测试方法。

4.7 数据管道

数据表不是声明测试用例的唯一方式,实际上,数据表只是数据管道的语法糖:

...
where:
a << [3, 7, 0]
b << [5, 0, 0]
c << [5, 7, 0]

数据管道用左 shift( << )操作符声明,左边是测试变量,右边是提供数据的管道。针对每个测试用例,所有管道都会提供一个数据给相应的测试变量。管道不一定就是数组,可以是任意的可迭代对象,比如文本文件,数据库表或随机数值生成器。

4.8 多值数据管道

多值数据管道会针对每个测试用例一次提供多个值给测试变量组:

@Shared sql = Sql.newInstance("jdbc:h2:mem:", "org.h2.Driver")

def "maximum of two numbers"() {
    ...
    where:
    [a, b, c] << sql.rows("select a, b, c from maxdata")
}

在管道提供的多个值中,如果有些值是无关紧要的,可以将对应位置的测试变量用占位符 _ 占用,也就是说,这些我们不关心的值,不会再用一个特定的测试变量去接收了,具体如下:

...
where:
[a, b, _, c] << sql.rows("select * from maxdata")

4.9 测试变量赋值

where 块中可以直接对测试变量进行赋值:

...
where:
a = 3
b = Math.random() * 100
c = a > b ? a : b

注意,测试变量赋值,数据管道和数据表这三种方式是可以混用的,针对每次测试用例,这些赋值操作都会执行一遍(也就是说会反复执行了)。赋值语句的右边可以出现其他测试变量:

...
where:
row << sql.rows("select * from maxdata")
// pick apart columns
a = row.a
b = row.b
c = row.c

4.10 数据表,数据管道和变量赋值混用

这三种方式可以根据需要混用:

...
where:
a | _
3 | _
7 | _
0 | _

b << [5, 0, 0]

c = a > b ? a : b

注意如果数据表或管道提供的数据数量不一致,则可能会引发异常。如果 where 块中只有变量赋值,则表明只有一个测试用例。数据表和数据管道在测试方法结束后会自动关闭,所以不需要在 cleanup 块或 cleanup() 中手动清理了。

5. 交互驱动测试

交互驱动测试关注的是被测试对象和外部协作者之间的交互行为,而不是被测试对象的内部状态。举个栗子,有一个消息推送者(被测对象),可以发送消息给若干消息订阅者(协作者):

class Publisher {
    List<Subscriber> subscribers
    void send(String message)
}

interface Subscriber {
    void receive(String message)
}

class PublisherSpec extends Specification {
    Publisher publisher = new Publisher()
}

我们要怎么测试推送者呢?在传统的状态驱动的测试中,我们只需要验证推送者内部保持了所有订阅者的引用即可,但是消息是否真的发出去了是否真的被接收到了,这个就无从得知了。为了验证这种消息传递的行为真的发生了,我们需要构建一种订阅者的特殊代理对象,这种对象就称为 Mock 对象。

其实我们可以完全手写这种代理,但是如果被代理对象很复杂很庞大,那手写就很不爽了。这时候就需要使用一些 mocking 框架,mocking 框架主要有三点作用:

  • 快速生成 mock 对象(一般使用 JDK 动态代理)

  • 定义被测试对象和协作者的预期交互行为

  • 验证定义好的预期交互行为

Java领域有很多 mocking 框架,比如 JMock, EasyMock, Mockito 等等。这些框架都可以和 Spock 协同使用,但更推荐使用 Spock 自带的 mocking 框架。

5.1 创建 Mock 对象

通过 MockingApi.Mock() 方法可以创建 Mock 对象:

def subscriber = Mock(Subscriber)
def subscriber2 = Mock(Subscriber)

还有另一种更推荐的写法:

// 自动推断要 mock 的对象的类型
Subscriber subscriber = Mock()
Subscriber subscriber2 = Mock()

Mock 对象的真实类型其实是被代理对象的子类型,这里其实是一种多态的写法。

5.2 Mock 对象的默认行为

调用 Mock 对象上的方法将返回相应类型的默认值,比如 0, false 或 null,只有 Object.toString,Object.hashCode 和 Object.equals 方法例外。Mock 对象的比较是对象地址的比较,toString 方法也会返回有内容的字符串来表明自己以及自己代理的对象。这些默认行为是可以覆盖的,如何覆盖将在 Stubbing 一节详述。

5.3 向被测对象注入 Mock 对象

将代理订阅者的 Mock 对象注入到推送者中:

class PublisherSpec extends Specification {
    Publisher publisher = new Publisher()
    Subscriber subscriber = Mock()
    Subscriber subscriber2 = Mock()

    def setup() {
        publisher.subscribers << subscriber // << 是 Groovy 的语法糖,List.add() 的简便写法
        publisher.subscribers << subscriber2
    }
}

接下来描述订阅者和推送者之间预期的交互行为。

5.4 Mocking

def "should send messages to all subscribers"() {
    when:
    publisher.send("hello")

    then:
    1 * subscriber.receive("hello")
    1 * subscriber2.receive("hello")
}

上面这段代码的意思就是:当推送者发送消息 hello,所有订阅者都将接收到一次且仅有一次消息 hello。

当 when 块中的代码运行的时候,Mock 对象上的方法调用情况必须要和 then 块中描述的完全一致,如果有不一致的地方,就会抛 InteractionNotSatisfiedError 异常。这种验证过程是自动进行的,不需要额外代码。

then 块中的交互描述语句可以分为四个部分,分别是:

1 * subscriber.receive("hello")
|   |          |       |
|   |          |       目标参数
|   |          目标方法
|   目标 Mock 对象
调用次数

5.4.1 调用次数

调用次数可以是一个具体值,也可以是范围:

1 * subscriber.receive("hello")      // 有且仅有一次调用
0 * subscriber.receive("hello")      // 0 次调用
(1..3) * subscriber.receive("hello") // 1 到 3 次调用,包括 1 和 3
(1.._) * subscriber.receive("hello") // 至少一次调用
(_..3) * subscriber.receive("hello") // 最多三次调用
_ * subscriber.receive("hello")      // 任意次数的调用,包括 0 次

5.4.2 目标 Mock 对象

1 * subscriber.receive("hello") // 调用发生在 'subscriber' 对象上
1 * _.receive("hello")          // 调用发生在任意 Mock 对象上

5.4.3 目标方法

1 * subscriber.receive("hello") // 'receive' 方法将会被调用
1 * subscriber./r.*e/("hello")  // 名字匹配正则表达式 r.*e 的方法将会被调用

如果目标方法是 getter 方法,可以简化为:

1 * subscriber.status // 等同于:1 * subscriber.getStatus()

如果目标方法是 setter 方法,就没有什么简化写法了:

1 * subscriber.setStatus("ok") // 不可以写成:1 * subscriber.status = "ok"

5.4.4 目标参数

1 * subscriber.receive("hello")     // 调用将会传入参数 "hello"
1 * subscriber.receive(!"hello")    // 调用将会传入一个不是 "hello" 的参数
1 * subscriber.receive()            // 调用不会传入任何参数
1 * subscriber.receive(_)           // 调用将会传入一个值任意的参数(包括 null)
1 * subscriber.receive(*_)          // 调用将会传入任意数量(包括 0),任意值的参数
1 * subscriber.receive(!null)       // 调用将会传入一个任意的非 null 的参数
1 * subscriber.receive(_ as String) // 调用将会传入一个 String 类型的非 null 的参数
1 * subscriber.receive({ it.size() > 3 }) // 调用将会传入一个参数,该参数能使 lambda 表达式返回 true
                                          // 这里是指传入参数的长度将会大于 3

以上这些花哨的目标参数描述是可以混用的:

1 * process.invoke("ls", "-a", _, !null, { ["abcdefghiklmnopqrstuwx1"].contains(it) })

5.4.5 匹配任意调用

1 * subscriber._(*_)     // 匹配 subscriber mock 对象上的任意方法,参数数量和值都任意
1 * subscriber._         // 上一行的简便写法,更推荐

1 * _._                  // 匹配任意 mock 对象上的任意方法,参数数量和值都任意
1 * _                    // 上一行的简便写法,更推荐

5.4.6 更严格的 Mocking

以上对交互行为的描述都是不严格的,为什么不严格呢?因为只描述了可以有某些行为,没有描述不可以有某些行为。严格的 mocking 应该指明,除了描述的行为外,其他任何行为都不应该发生:

when:
publisher.publish("hello")

then:
1 * subscriber.receive("hello") // 期望 'subscriber' mock 对象上的一次 'receive' 调用
_ * auditing._                  // 'auditing' mock 对象上的任何调用都是允许的
0 * _                           // 除上面两条描述以外的其他任何行为都是不允许的,该描述必须是最后一条描述

5.4.7 交互描述语句的位置

交互描述语句除了可以放在 then 块,放在 when 块之前的所有地方都是可以的,比如 setup 块。此外,交互描述语句放在辅助方法中也是可以的。

如果一次调用匹配多个交互描述,那么匹配最先定义的调用次数没有用完的交互描述语句,而且优先从 then 块开始匹配。

5.4.8 创建 mock 对象时描述交互行为

如果 mock 对象有一些基本的通用的交互行为,不太可能根据上下文发生变化的,则可以在创建该 mock 对象时就定义:

def subscriber = Mock(Subscriber) {
   1 * receive("hello")
   1 * receive("goodbye")
}

也可以这么写:

class MySpec extends Specification {
    Subscriber subscriber = Mock {
        1 * receive("hello")
        1 * receive("goodbye")
    }
}

5.4.9 对交互进行分组

如果有大量的交互描述语句,它们的目标 mock 对象都是同一个,那么可以将这些交互描述语句分为一个组:

// 省去了反复写 subscriber 的麻烦
with(subscriber) {
    1 * receive("hello")
    1 * receive("goodbye")
}

5.4.10 交互描述语句和条件语句混合

then 块中可以同时包括交互描述语句和条件语句,且一般来说,会把交互描述语句放在条件语句前面:

when:
publisher.send("hello")

then:
1 * subscriber.receive("hello")
publisher.messageCount == 1

5.4.11 显式交互描述代码块

实际上,spock 框架会在 when 块执行之前执行 then 块中的所有交互描述语句(类似于变量的先定义才能使用)。一般情况下,这种颠倒的执行顺序不会出错,但是下面这种情况就有问题了:

when:
publisher.send("hello")

then:
def message = "hello" // 1
1 * subscriber.receive(message) // 2

执行顺序是先2再when最后1。2中的目标参数是一个变量(也完全可以让调用次数成为变量),在这种颠倒的执行顺序下,message变量明显找不到定义。一种解决办法是将message定义在 when 块前或 where 块中,另一种办法就是将变量定义语句和交互描述语句绑定:

when:
publisher.send("hello")

then:
interaction { // 作为整体执行
    def message = "hello"
    1 * subscriber.receive(message)
}

5.4.12 交互描述语句的作用域

如果交互在 then 块中定义,那么它的作用域就是对应的 where 块:

when:
publisher.send("message1")

then:
1 * subscriber.receive("message1")

when:
publisher.send("message2")

then:
1 * subscriber.receive("message2")

上面的代码确保了在第一个 when 块执行的时候,subscriber mock 对象会接收到消息message1,在第二个 when 块执行的时候会接收到消息message2。

定义在 then 块外的交互的作用域是从它的定义开始,直到包含它的测试方法结束。

交互只能在测试方法中定义,不能定义在静态方法,setupSpec 方法和 cleanupSpec 方法中。一样的道理,mock 对象也不能作为静态字段和 @Shared 字段。

5.4.13 交互验证

交互驱动的测试失败主要有两种原因:实际的调用次数大于期望的调用次数,或实际的调用次数小于期望的调用次数。对于前者,一旦调用次数超出,就会立即抛出 TooManyInvocationsError 错误:

Too many invocations for:

2 * subscriber.receive(_) (3 invocations)

为了更容易诊断问题的原因,spock 会列出已经发生的所有匹配的调用:

Matching invocations (ordered by last occurrence):

2 * subscriber.receive("hello")   <-- this triggered the error
1 * subscriber.receive("goodbye")

根据输出,第二次 receive("hello") 调用引发了 TooManyInvocationsError 错误。注意到两次 receive("hello") 调用实际上是难以区分的,所以就把两次调用合并为了一行。第一次 receive("hello") 调用是有可能先于 receive("goodbye") 的。

第二种情况(调用次数过少)只能在 when 块完成后检测到。下面的输出就表示出现了调用次数过少的错误(TooFewInvocationsError):

Too few invocations for:

1 * subscriber.receive("hello") (0 invocations)

引发这种错误的更进一步的原因可能有:

  • 调用该方法的时候参数不对

  • 在其他 mock 对象上调用的该方法

  • 调用了该 mock 对象上的其他方法,而不是这个 receive 方法

不管是上面的哪种情况,TooFewInvocationsError 错误都会引发。

为了更好地诊断调用次数过少的真正原因,spock 会将没有匹配任何交互的调用全部列出,并根据它们和出错交互的相似程度排序。比如现有一次调用,和出错交互很像,只有参数没对上,那么这次调用就会最先列出:

Unmatched invocations (ordered by similarity):

1 * subscriber.receive("goodbye") // 和 1 * subscriber.receive("hello") 最相似
1 * subscriber2.receive("hello")

5.4.14 调用次序

虽然交互描述语句的定义是有顺序的,但实际发生的调用可以是任意顺序。比如:

then:
2 * subscriber.receive("hello")
1 * subscriber.receive("goodbye")

实际调用的顺序可以是"hello", "hello", "goodbye",可以是"hello", "goodbye", "hello",也可以是"goodbye", "hello", "hello",都是可以通过测试的,通过下面的方式可以指定调用的顺序:

then:
2 * subscriber.receive("hello")

then:
1 * subscriber.receive("goodbye")

指定 receive 方法的前两次调用传入的参数是"hello",第三次调用传入的参数是"goodbye"。

5.4.15 mocking 类

到目前为止,所有的 mocking 都是在 mocking 接口。其实 mocking 类和 mocking 接口是完全一样的,唯一的区别是 mocking 类需要额外的依赖:cglib-nodep-2.2 以上版本和 objenesis-1.2 以上版本。两者缺一不可,如果缺少,spock 框架会提示。

5.5 Stubbing

Stubbing 只关心调用的返回值,不关心调用发生了多少遍。可以通过 stubbing 覆盖掉 mock 对象的默认行为。举个栗子:

interface Subscriber {
    String receive(String message)
}

现在指定 receive 方法的每次调用都返回 "ok":

subscriber.receive(_) >> "ok"

相比交互描述语句,少了调用次数,最右边多了返回值生成器:

subscriber.receive(_) >> "ok"
|          |       |     |
|          |       |     返回值生成器
|          |       目标参数
|          目标方法
目标 mock 对象

stubbed 交互可以定义在 then 块中,也可以定义在 when 块前。如果一个 mock 对象只是用来返回一些值的,那一般来说会把相关的 stubbed 交互定义在 given 块,或创建该 mock 对象的时候。

5.5.1 返回固定值

和刚才的操作一样,使用右位移操作符(>>)指定交互返回一个固定值:

subscriber.receive(_) >> "ok"

对不同的调用需要返回不同的值,可以这么做:

subscriber.receive("message1") >> "ok"
subscriber.receive("message2") >> "fail"

接收到消息 "message1" 的时候,指定返回 "ok",接收到消息 "message2" 的时候,指定返回 "fail",当然也可以指定其他任意值,只要和 receive 方法的返回类型兼容即可。

5.5.2 返回若干数量的值

为了便捷地在一系列调用中指定不同的返回值,可以使用 >>> 操作符:

subscriber.receive(_) >>> ["ok", "error", "error", "ok"]

第一次调用指定返回 "ok",第二次和第三次调用指定返回 "error",第四次调用指定返回 "ok"。>>> 操作符的右边必须是一个 groovy 知道如何迭代的可迭代对象,在上面的栗子里,是一个 groovy 数组。

5.5.3 根据参数计算返回值

为了根据方法参数动态指定返回值,可以在 >> 右边跟一个闭包,如果闭包有一个无类型的参数,那么该参数就代表方法的参数列表:

subscriber.receive(_) >> { args -> args[0].size() > 3 ? "ok" : "fail" }

如果第一个参数(只有一个参数)的长度(当前上下文中应该是指字符数)大于3,返回 "ok",否则返回 "fail"。

在多数情况下,如果能在闭包中直接访问方法参数会方便很多。如果闭包有多个参数,或有一个带类型的参数,那么方法参数会映射到闭包参数上:

subscriber.receive(_) >> { String message -> message.size() > 3 ? "ok" : "fail" }

返回值生成器的行为和前一种写法完全一样,但可读性好了很多。

5.5.4 抛出异常

除了指定返回值,也可以指定抛出异常:

subscriber.receive(_) >> { throw new InternalError("ouch") }

当然闭包里也可以加其他操作,比如打印日志之类的,每当调用发生的时候都会执行。

5.5.5 链式返回值生成器

subscriber.receive(_) >>> ["ok", "fail", "ok"] >> { throw new InternalError() } >> "ok"

前三次调用依次返回 "ok","fail","ok",第四次调用抛出 InternalError 异常,第五次及以后的调用都返回 "ok"。

5.6 结合 Mocking 和 Stubbing

mocking 和 stubbing 可以按如下方式结合:

1 * subscriber.receive("message1") >> "ok" // 不可拆分
1 * subscriber.receive("message2") >> "fail"

5.7 其他

关于交互驱动测试 spock 还提供了其他特性,比如 mock 构造器,mock 静态方法等。但是不推荐使用,如果真的需要去用这些特性,说明你的代码设计的有问题,更好的做法是重构你的业务代码,而不是去用这些冷门的特性。

6. 进阶扩展注解

6.1 IgnoreIf

在特定条件下忽略被注解的测试方法,该注解必须传入一个 predicate 类型的闭包:

@IgnoreIf({ System.getProperty("os.name").contains("windows") })
def "I'll run everywhere but on Windows"() { ... }

为了使可读性更强,写起来更方便,可以使用一些闭包中内置的变量:

  • sys 包含所有 system 属性,键值对形式

  • env 包含所有环境变量

  • os 包含操作系统信息,spock.util.environment.OperatingSystem 类型

  • jvm 包含 JVM 信息,spock.util.environment.Jvm 类型

使用 os 变量,上面的代码可以简化为这个样子:

@IgnoreIf({ os.windows })
def "I'll run everywhere but on Windows"() { ... }

使用 @Ignore 和 @IgnoreIf 的时候要注意,如果测试类上有 @StepWise,那么忽略掉一些测试方法可能会对后面的测试方法产生影响。

6.2 Requires

@Requires 和 @IgnoreIf 很像,只不过 @IgnoreIf 是在 predicate 闭包返回 true 时不执行测试方法,而 @Requires 是在 predicate 闭包返回 false 时不执行测试方法,正好相反。

@Requires({ os.windows })
def "I'll only run on Windows"() { ... }

6.3 Retry

@Retry 一般在进行一些小的集成测试的时候会用到,比如说测试方法中真的要去调用远程服务(直接 mock 掉叫单元测试,真的去做远程调用就是集成测试了)。这个注解有5个配置项:

  • 重试次数,默认3次

  • 每次重试之间的时间间隔,默认0,默认单位毫秒

  • 测试方法出现了何种未处理异常才会触发重试

  • 触发重试的条件,用 predicate 闭包定义

  • 对于数据驱动测试,有两种重试模式可供配置,仅重试发生错误的那次迭代,或重试整个测试方法

class FlakyIntegrationSpec extends Specification {
  @Retry
  def retry3Times() { ... }

  @Retry(count = 5)
  def retry5Times() { ... }

  @Retry(exceptions=[IOException])
  def onlyRetryIOException() { ... }

  @Retry(condition = { failure.message.contains('foo') })
  def onlyRetryIfConditionOnFailureHolds() { ... }

  @Retry(condition = { instance.field != null })
  def onlyRetryIfConditionOnInstanceHolds() { ... }

  @Retry
  def retryFailingIterations() {
    ...
    where:
    data << sql.select()
  }

  @Retry(mode = Retry.Mode.FEATURE)
  def retryWholeFeature() {
    ...
    where:
    data << sql.select()
  }

  @Retry(delay = 1000)
  def retryAfter1000MsDelay() { ... }
}

@Retry 也可以标记测试类,相当于标记类中的所有测试方法。测试方法上的 @Retry 会覆盖测试类上的 @Retry。

@Retry
class FlakyIntegrationSpec extends Specification {
  def "will be retried with config from class"() {
    ...
  }
  @Retry(count = 5)
  def "will be retried using its own config"() {
    ...
  }
}

7. 与spring-boot集成

7.1 maven

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-test</artifactId>
    <scope>test</scope>
</dependency>
<dependency>
    <groupId>org.spockframework</groupId>
    <artifactId>spock-core</artifactId>
    <scope>test</scope>
</dependency>
<dependency>
    <groupId>org.spockframework</groupId>
    <artifactId>spock-spring</artifactId>
    <scope>test</scope>
</dependency>
<dependency>
    <groupId>cglib</groupId>
    <artifactId>cglib-nodep</artifactId>
    <version>3.2.9</version>
    <scope>test</scope>
</dependency>

7.2 测试spring-boot接口

@SpringBootTest(webEnvironment = SpringBootTest.WebEnvironment.RANDOM_PORT)
@ContextConfiguration(classes = Application) // 指定入口点
@ActiveProfiles("test")
@Stepwise
class XxxControllerSpec extends Specification {
  @Autowired
  private TestRestTemplate restTemplate
  // 发起请求restTemplate.exchange(url, method, headers, responseType)
}
2021-04-15-aWUrge
Spock官方文档翻译(非机翻)
http://spockframework.org/spo...