chore: sync with upstream e7cb442 + update zh translations

This commit is contained in:
xuxiang
2026-02-02 18:57:56 +08:00
parent 6f87d43c19
commit d7cafbe582
66 changed files with 9395 additions and 1465 deletions

View File

@@ -1,91 +1,91 @@
---
name: java-coding-standards
description: Java coding standards for Spring Boot services: naming, immutability, Optional usage, streams, exceptions, generics, and project layout.
description: Spring Boot 服务的 Java 编码规范命名、不可变性Immutability)、Optional 使用、流Streams、异常处理、泛型Generics和项目布局。
---
# Java Coding Standards
# Java 编码规范
Standards for readable, maintainable Java (17+) code in Spring Boot services.
适用于 Spring Boot 服务中易读、可维护的 Java (17+) 代码规范。
## Core Principles
## 核心原则
- Prefer clarity over cleverness
- Immutable by default; minimize shared mutable state
- Fail fast with meaningful exceptions
- Consistent naming and package structure
- 清晰胜过奇巧
- 默认不可变(Immutable);尽量减少共享的可变状态
- 快速失败并抛出有意义的异常
- 保持一致的命名和包结构
## Naming
## 命名
```java
// ✅ Classes/Records: PascalCase
// ✅ 类(Classes/ 记录(Records):大驼峰式(PascalCase
public class MarketService {}
public record Money(BigDecimal amount, Currency currency) {}
// ✅ Methods/fields: camelCase
// ✅ 方法/字段:小驼峰式(camelCase
private final MarketRepository marketRepository;
public Market findBySlug(String slug) {}
// ✅ Constants: UPPER_SNAKE_CASE
// ✅ 常量:大写下划线命名式(UPPER_SNAKE_CASE
private static final int MAX_PAGE_SIZE = 100;
```
## Immutability
## 不可变性(Immutability
```java
// ✅ Favor records and final fields
// ✅ 优先使用记录Records)和 final 字段
public record MarketDto(Long id, String name, MarketStatus status) {}
public class Market {
private final Long id;
private final String name;
// getters only, no setters
// 仅提供 getter,不提供 setter
}
```
## Optional Usage
## Optional 使用
```java
// ✅ Return Optional from find* methods
// ✅ find* 方法返回 Optional
Optional<Market> market = marketRepository.findBySlug(slug);
// ✅ Map/flatMap instead of get()
// ✅ 使用 map/flatMap 而不是 get()
return market
.map(MarketResponse::from)
.orElseThrow(() -> new EntityNotFoundException("Market not found"));
```
## Streams Best Practices
## 流(Streams)最佳实践
```java
// ✅ Use streams for transformations, keep pipelines short
// ✅ 使用流进行转换,保持流水线简短
List<String> names = markets.stream()
.map(Market::name)
.filter(Objects::nonNull)
.toList();
// ❌ Avoid complex nested streams; prefer loops for clarity
// ❌ 避免复杂的嵌套流;为了清晰起见,优先使用循环
```
## Exceptions
## 异常处理
- Use unchecked exceptions for domain errors; wrap technical exceptions with context
- Create domain-specific exceptions (e.g., `MarketNotFoundException`)
- Avoid broad `catch (Exception ex)` unless rethrowing/logging centrally
- 领域错误使用非受检异常Unchecked Exceptions为技术异常包装上下文信息
- 创建领域特定的异常(例如 `MarketNotFoundException`
- 避免宽泛的 `catch (Exception ex)`除非是进行集中式重抛Rethrow或日志记录
```java
throw new MarketNotFoundException(slug);
```
## Generics and Type Safety
## 泛型与类型安全
- Avoid raw types; declare generic parameters
- Prefer bounded generics for reusable utilities
- 避免原始类型Raw Types);声明泛型参数
- 可复用工具类优先使用有界泛型Bounded Generics
```java
public <T extends Identifiable> Map<Long, T> indexById(Collection<T> items) { ... }
```
## Project Structure (Maven/Gradle)
## 项目结构 (Maven/Gradle)
```
src/main/java/com/example/app/
@@ -98,25 +98,25 @@ src/main/java/com/example/app/
util/
src/main/resources/
application.yml
src/test/java/... (mirrors main)
src/test/java/... (结构与 main 保持镜像)
```
## Formatting and Style
## 格式与风格
- Use 2 or 4 spaces consistently (project standard)
- One public top-level type per file
- Keep methods short and focused; extract helpers
- Order members: constants, fields, constructors, public methods, protected, private
- 一致地使用 2 4 个空格(遵循项目标准)
- 每个文件仅包含一个公共顶层类型
- 保持方法简短且聚焦提取辅助方法Helper methods
- 成员排序:常量、字段、构造函数、公共方法、受保护方法、私有方法
## Code Smells to Avoid
## 需避免的代码坏味道(Code Smells
- Long parameter lists → use DTO/builders
- Deep nesting → early returns
- Magic numbers → named constants
- Static mutable state → prefer dependency injection
- Silent catch blocks → log and act or rethrow
- 长参数列表 → 使用 DTO/构建器Builders
- 深层嵌套 → 尽早返回Early returns
- 魔法数字 → 具名常量
- 静态可变状态 → 优先使用依赖注入Dependency Injection
- 静默的 catch 块 → 记录日志并处理或重抛
## Logging
## 日志记录
```java
private static final Logger log = LoggerFactory.getLogger(MarketService.class);
@@ -124,15 +124,15 @@ log.info("fetch_market slug={}", slug);
log.error("failed_fetch_market slug={}", slug, ex);
```
## Null Handling
## 空值处理(Null Handling
- Accept `@Nullable` only when unavoidable; otherwise use `@NonNull`
- Use Bean Validation (`@NotNull`, `@NotBlank`) on inputs
- 仅在不可避免时接受 `@Nullable`;否则使用 `@NonNull`
- 在输入上使用 Bean 校验(`@NotNull`, `@NotBlank`
## Testing Expectations
## 测试预期
- JUnit 5 + AssertJ for fluent assertions
- Mockito for mocking; avoid partial mocks where possible
- Favor deterministic tests; no hidden sleeps
- 使用 JUnit 5 + AssertJ 进行流式断言Fluent Assertions
- 使用 Mockito 进行模拟Mocking尽可能避免部分模拟Partial Mocks
- 倾向于确定性测试;不要包含隐藏的 sleep 操作
**Remember**: Keep code intentional, typed, and observable. Optimize for maintainability over micro-optimizations unless proven necessary.
**记住**:保持代码的意图清晰、类型安全且具有可观测性。除非证明有必要,否则应优先考虑可维护性而非微优化。