Files

3.9 KiB
Raw Permalink Blame History

name, description
name description
java-coding-standards Spring Boot 服务的 Java 编码规范命名、不可变性Immutability、Optional 使用、流Streams、异常处理、泛型Generics和项目布局。

Java 编码规范

适用于 Spring Boot 服务中易读、可维护的 Java (17+) 代码规范。

核心原则

  • 清晰胜过奇巧
  • 默认不可变Immutable尽量减少共享的可变状态
  • 快速失败并抛出有意义的异常
  • 保持一致的命名和包结构

命名

// ✅ 类Classes/ 记录Records大驼峰式PascalCase
public class MarketService {}
public record Money(BigDecimal amount, Currency currency) {}

// ✅ 方法/字段小驼峰式camelCase
private final MarketRepository marketRepository;
public Market findBySlug(String slug) {}

// ✅ 常量大写下划线命名式UPPER_SNAKE_CASE
private static final int MAX_PAGE_SIZE = 100;

不可变性Immutability

// ✅ 优先使用记录Records和 final 字段
public record MarketDto(Long id, String name, MarketStatus status) {}

public class Market {
  private final Long id;
  private final String name;
  // 仅提供 getter不提供 setter
}

Optional 使用

// ✅ find* 方法返回 Optional
Optional<Market> market = marketRepository.findBySlug(slug);

// ✅ 使用 map/flatMap 而不是 get()
return market
    .map(MarketResponse::from)
    .orElseThrow(() -> new EntityNotFoundException("Market not found"));

Streams最佳实践

// ✅ 使用流进行转换,保持流水线简短
List<String> names = markets.stream()
    .map(Market::name)
    .filter(Objects::nonNull)
    .toList();

// ❌ 避免复杂的嵌套流;为了清晰起见,优先使用循环

异常处理

  • 领域错误使用非受检异常Unchecked Exceptions为技术异常包装上下文信息
  • 创建领域特定的异常(例如 MarketNotFoundException
  • 避免宽泛的 catch (Exception ex)除非是进行集中式重抛Rethrow或日志记录
throw new MarketNotFoundException(slug);

泛型与类型安全

  • 避免原始类型Raw Types声明泛型参数
  • 可复用工具类优先使用有界泛型Bounded Generics
public <T extends Identifiable> Map<Long, T> indexById(Collection<T> items) { ... }

项目结构 (Maven/Gradle)

src/main/java/com/example/app/
  config/
  controller/
  service/
  repository/
  domain/
  dto/
  util/
src/main/resources/
  application.yml
src/test/java/... (结构与 main 保持镜像)

格式与风格

  • 一致地使用 2 或 4 个空格(遵循项目标准)
  • 每个文件仅包含一个公共顶层类型
  • 保持方法简短且聚焦提取辅助方法Helper methods
  • 成员排序:常量、字段、构造函数、公共方法、受保护方法、私有方法

需避免的代码坏味道Code Smells

  • 长参数列表 → 使用 DTO/构建器Builders
  • 深层嵌套 → 尽早返回Early returns
  • 魔法数字 → 具名常量
  • 静态可变状态 → 优先使用依赖注入Dependency Injection
  • 静默的 catch 块 → 记录日志并处理或重抛

日志记录

private static final Logger log = LoggerFactory.getLogger(MarketService.class);
log.info("fetch_market slug={}", slug);
log.error("failed_fetch_market slug={}", slug, ex);

空值处理Null Handling

  • 仅在不可避免时接受 @Nullable;否则使用 @NonNull
  • 在输入上使用 Bean 校验(@NotNull, @NotBlank

测试预期

  • 使用 JUnit 5 + AssertJ 进行流式断言Fluent Assertions
  • 使用 Mockito 进行模拟Mocking尽可能避免部分模拟Partial Mocks
  • 倾向于确定性测试;不要包含隐藏的 sleep 操作

记住:保持代码的意图清晰、类型安全且具有可观测性。除非证明有必要,否则应优先考虑可维护性而非微优化。