96SEO 2026-02-19 23:00 8
单元测试实战#xff08;一#xff09;Controller

单元测试实战特分门别类示例各种组件的测试代码并进行解说供开发人员参考。
框架都提供根据一定规约自动生成查询/更新的操作如代理接口。
这一部分责任在框架本身我们不需测试。
因此DAO的测试应主要针对自定义查询/更新操作。
JPA的测试注解是DataJpaTestMybatis-Plus的是MyBatisPlusTest它们的思想是一致的此次先讲JPA。
JPA标准的Repository且使用接口代理那么理论上是不需要测试的但实际中不排除想验证订制代码native
boot测试框架提供的功能因此它是有Spring上下文的使用JUnit的SpringExtension扩展类。
测试还是遵循经典三段式given、when、***n即假设xxx……那么当yyy时……应该会zzz。
dependencygroupIdorg.springframework.boot/groupIdartifactIdspring-boot-starter-test/artifactIdscopetest/scope
dependencygroupIdorg.junit.jupiter/groupIdartifactIdjunit-jupiter-api/artifactIdscopetest/scope
dependencygroupIdcom.h2database/groupIdartifactIdh2/artifactIdscopetest/scope
com.aaa.api.auth.repository;import
org.springframework.data.jpa.repository.JpaRepository;
org.springframework.stereotype.Repository;import
findTop10ByNameContaining(String
上面的Repository虽然没有订制代码但不妨碍我们用它来演示Repository测试的写法。
com.aaa.api.auth.repository;import
org.junit.jupiter.api.BeforeEach;
org.springframework.beans.factory.annotation.Autowired;
org.springframework.boot.test.autoconfigure.orm.jpa.DataJpaTest;
org.springframework.boot.test.autoconfigure.orm.jpa.TestEntityManager;
org.springframework.test.context.TestPropertySource;import
org.assertj.core.api.Assertions.assertThat;DataJpaTest
//AutoConfigureTestDatabase(replace
AutoConfigureTestDatabase.Replace.NONE)
{//spring.datasource.driver-class-nameorg.h2.Driver,//spring.datasource.urljdbc:h2:mem:api_auth_test;DB_CLOSE_DELAY-1;DB_CLOSE_ON_EXITFALSE;MODEMySQL;DATABASE_TO_LOWERTRUE;NON_KEYWORDSuser,//spring.datasource.usernamesa,spring.jpa.hibernate.ddl-autocreate-drop,spring.jpa.properties.hibernate.hbm2ddl.autocreate-drop,spring.jpa.properties.hibernate.globally_quoted_identifierstrue
{u1.setName(张三);u1.setUserCode(zhangsan);u1.setRole(User.ADMIN);u1.setEmail(zhangsanaaa.net.cn);u1.setMobile(13600001234);u2.setName(李四);u2.setUserCode(lisi);u2.setRole(User.ADMIN);u2.setEmail(lisiaaa.net.cn);u2.setMobile(13800001234);u3.setName(王五);u3.setUserCode(wangwu);u3.setRole(User.USER);u3.setEmail(wangwuaaa.net.cn);u3.setMobile(13900001234);}Testvoid
entityManager.getEntityManager().createQuery(from
User);repo.save(u1);assertThat(q.getResultList()).hasSize(1);repo.save(u2);assertThat(q.getResultList()).hasSize(2);User
repo.save(u3);assertThat(q.getResultList()).hasSize(3);u3.setRole(User.ADMIN);repo.save(u3);assertThat(q.getResultList()).hasSize(3);assertThat(entityManager.find(User.class,
u.getId()).getRole()).isEqualTo(User.ADMIN);}
{entityManager.persist(u1);entityManager.persist(u2);entityManager.persist(u3);ListUser
repo.findAll();assertThat(entities).size().isEqualTo(3);assertThat(entities.get(0).getId()).isNotNull();assertThat(entities.get(1).getId()).isNotNull();assertThat(entities.get(2).getId()).isNotNull();}Testvoid
{entityManager.persist(u1);entityManager.persist(u2);entityManager.persist(u3);User
repo.findByUserCode(zhangsan).orElse(null);assertThat(u1).isNotNull();assertThat(u1.getRole()).isEqualTo(User.ADMIN);User
repo.findByUserCode(lisi).orElse(null);assertThat(u2).isNotNull();assertThat(u2.getRole()).isEqualTo(User.ADMIN);User
repo.findByUserCode(wangwu).orElse(null);assertThat(u3).isNotNull();assertThat(u3.getRole()).isEqualTo(User.USER);u3.setRole(User.ADMIN);entityManager.persist(u3);u3
repo.findByUserCode(wangwu).orElse(null);assertThat(u3).isNotNull();assertThat(u3.getRole()).isEqualTo(User.ADMIN);}Testvoid
testFindTop10ByNameContaining()
{entityManager.persist(u2);entityManager.persist(u3);ListUser
repo.findTop10ByNameContaining(张三);assertThat(users).isEmpty();users
repo.findTop10ByNameContaining(李四);assertThat(users).hasSize(1);users
repo.findTop10ByNameContaining(五);assertThat(users).hasSize(1);}}
DataJpaTest、WebMvcTest以及SpringBootTest等测试其实是需要Spring
Boot的Configuration的因此我们在测试目录里可能尤其在多模块工程里有个专用于测试的Mock
Application注解为SpringBootApplication空类不需要main方法并且在src/test/resources下还可以有application.properties或yaml。
而更灵活的方式是使用第18行的TestPropertySource的properties属性。
DataJpaTest这个注解会在测试时自动将数据源替换为内存数据库H2的如果不想要这种替换或者要订制其url等属性那么就可以将第17行、第19-21行的注释放开。
第24行的spring.jpa.properties.hibernate.globally_quoted_identifierstrue可以避免数据库保留字与表名/列名冲突。
第29行我们将待测试类对象作为测试类的一个属性并使用Autowired进行注入。
第32行我们注入了一个TestEntityManager它作为一个实用工具帮我们在测试中插数据、查数据这样就避免了直接使用repo的方法插数据、查数据因为这些方法本身就是待测目标。
第34-36行提供了三个测试数据并在setUp()方法中进行初始化/重置。
BeforeEach注解使得setUp()方法在每个测试之前都会执行一遍。
从59行开始是测试方法。
该测试类中没有任何Mock对象因此也就不存在given
***n三段式结构。
测试方法都是直接对测试数据进行CRUD操作并检查操作结果代码都是自解释的。
Repository的测试推荐使用DataJpaTest注解。
如需订制内存数据库datasource的属性则令AutoConfigureTestDatabase(replace
AutoConfigureTestDatabase.Replace.NONE)然后在TestPropertySource.properties里写订制属性。
Repository应设计为单纯的对数据实体的CRUD操作因此通常不需Mock任何依赖对象。
作为专业的SEO优化服务提供商,我们致力于通过科学、系统的搜索引擎优化策略,帮助企业在百度、Google等搜索引擎中获得更高的排名和流量。我们的服务涵盖网站结构优化、内容优化、技术SEO和链接建设等多个维度。
| 服务项目 | 基础套餐 | 标准套餐 | 高级定制 |
|---|---|---|---|
| 关键词优化数量 | 10-20个核心词 | 30-50个核心词+长尾词 | 80-150个全方位覆盖 |
| 内容优化 | 基础页面优化 | 全站内容优化+每月5篇原创 | 个性化内容策略+每月15篇原创 |
| 技术SEO | 基本技术检查 | 全面技术优化+移动适配 | 深度技术重构+性能优化 |
| 外链建设 | 每月5-10条 | 每月20-30条高质量外链 | 每月50+条多渠道外链 |
| 数据报告 | 月度基础报告 | 双周详细报告+分析 | 每周深度报告+策略调整 |
| 效果保障 | 3-6个月见效 | 2-4个月见效 | 1-3个月快速见效 |
我们的SEO优化服务遵循科学严谨的流程,确保每一步都基于数据分析和行业最佳实践:
全面检测网站技术问题、内容质量、竞争对手情况,制定个性化优化方案。
基于用户搜索意图和商业目标,制定全面的关键词矩阵和布局策略。
解决网站技术问题,优化网站结构,提升页面速度和移动端体验。
创作高质量原创内容,优化现有页面,建立内容更新机制。
获取高质量外部链接,建立品牌在线影响力,提升网站权威度。
持续监控排名、流量和转化数据,根据效果调整优化策略。
基于我们服务的客户数据统计,平均优化效果如下:
我们坚信,真正的SEO优化不仅仅是追求排名,而是通过提供优质内容、优化用户体验、建立网站权威,最终实现可持续的业务增长。我们的目标是与客户建立长期合作关系,共同成长。
Demand feedback