java-dev

参考来源: Google Java Style Guide、阿里巴巴 Java 开发手册

Safety Notice

This listing is imported from skills.sh public index metadata. Review upstream SKILL.md and repository scripts before running.

Copy this and send it to your AI assistant to learn

Install skill "java-dev" with this command: npx skills add doccker/cc-use-exp/doccker-cc-use-exp-java-dev

Java 开发规范

参考来源: Google Java Style Guide、阿里巴巴 Java 开发手册

工具链

Maven

mvn clean compile # 编译 mvn test # 运行测试 mvn verify # 运行所有检查

Gradle

./gradlew build # 构建 ./gradlew test # 运行测试

命名约定

类型 规则 示例

包名 全小写,域名反转 com.example.project

类名 大驼峰,名词/名词短语 UserService , HttpClient

方法名 小驼峰,动词开头 findById , isValid

常量 全大写下划线分隔 MAX_RETRY_COUNT

布尔返回值 is/has/can 前缀 isActive() , hasPermission()

类成员顺序

public class Example { // 1. 静态常量 public static final String CONSTANT = "value";

// 2. 静态变量
private static Logger logger = LoggerFactory.getLogger(Example.class);

// 3. 实例变量
private Long id;

// 4. 构造函数
public Example() { }

// 5. 静态方法
public static Example create() { return new Example(); }

// 6. 实例方法(公共 → 私有)
public void doSomething() { }
private void helperMethod() { }

// 7. getter/setter(或使用 Lombok)

}

DTO/VO 类规范

规则 说明

❌ 禁止手写 getter/setter DTO、VO、Request、Response 类一律使用 Lombok

✅ 使用 @Data

普通 DTO

✅ 使用 @Value

不可变 DTO

✅ 使用 @Builder

字段较多时配合使用

⚠️ Entity 类慎用 @Data

JPA Entity 的 equals/hashCode 会影响 Hibernate 代理

// ❌ 手写 getter/setter public class UserDTO { private Long id; private String name; public Long getId() { return id; } public void setId(Long id) { this.id = id; } // ... 大量样板代码 }

// ✅ 使用 Lombok @Data public class UserDTO { private Long id; private String name; }

批量查询规范

规则 说明

❌ 禁止 IN 子句超过 500 个参数 SQL 解析开销大,执行计划不稳定

✅ 超过时分批查询 每批 500,合并结果

✅ 封装通用工具方法 避免每处手写分批逻辑

// ❌ 1700 个 ID 一次查询 List<User> users = userRepository.findByIdIn(allIds); // IN 子句过长

// ✅ 分批查询工具方法 public static <T, R> List<R> batchQuery(List<T> params, int batchSize, Function<List<T>, List<R>> queryFn) { List<R> result = new ArrayList<>(); for (int i = 0; i < params.size(); i += batchSize) { List<T> batch = params.subList(i, Math.min(i + batchSize, params.size())); result.addAll(queryFn.apply(batch)); } return result; }

// 使用 List<User> users = batchQuery(allIds, 500, ids -> userRepository.findByIdIn(ids));

N+1 查询防范

规则 说明

❌ 禁止循环内调用 Repository/Mapper stream/forEach/for 内每次迭代触发一次查询

✅ 循环外批量查询,结果转 Map 查询次数从 N 降为 1(或 distinct 数)

// ❌ N+1:循环内逐行查询 count records.forEach(record -> { long count = deviceRepo.countByDeviceId(record.getDeviceId()); // 每条触发一次查询 record.setDeviceCount(count); });

// ✅ 循环外批量查询 + Map 查找 List<String> deviceIds = records.stream() .map(Record::getDeviceId).distinct().collect(Collectors.toList()); Map<String, Long> countMap = deviceRepo.countByDeviceIdIn(deviceIds).stream() .collect(Collectors.toMap(CountDTO::getDeviceId, CountDTO::getCount)); records.forEach(r -> r.setDeviceCount(countMap.getOrDefault(r.getDeviceId(), 0L)));

常见 N+1 场景及修复模式:

场景 循环内(❌) 循环外(✅)

count repo.countByXxx(id)

repo.countByXxxIn(ids) → Map<id, count>

findById repo.findById(id)

repo.findByIdIn(ids) → Map<id, entity>

exists repo.existsByXxx(id)

repo.findXxxIn(ids) → Set<id>

  • set.contains()

并发安全规范

规则 说明

❌ 禁止 read-modify-write 先读余额再写回,并发下丢失更新

❌ 禁止 check-then-act 无兜底 先检查再操作,并发下条件失效

✅ 使用原子更新 SQL UPDATE SET balance = balance + :delta WHERE id = :id

✅ 或使用乐观锁 @Version 字段 + 重试机制

✅ 唯一索引兜底 防重复插入的最后防线

// ❌ read-modify-write 竞态条件 PointsAccount account = accountRepo.findById(id); account.setBalance(account.getBalance() + points); // 并发时丢失更新 accountRepo.save(account);

// ✅ 方案一:原子更新 SQL @Modifying @Query("UPDATE PointsAccount SET balance = balance + :points WHERE id = :id") int addBalance(@Param("id") Long id, @Param("points") int points);

// ✅ 方案二:乐观锁 @Version private Long version; // Entity 中添加版本字段

// ❌ check-then-act 无兜底(并发下可能重复结算) if (!rewardRepo.existsByTenantIdAndPeriod(tenantId, period)) { rewardRepo.save(new RankingReward(...)); }

// ✅ 唯一索引兜底 + 异常捕获 // DDL: UNIQUE INDEX uk_tenant_period (tenant_id, ranking_type, period, rank_position) try { rewardRepo.save(new RankingReward(...)); } catch (DataIntegrityViolationException e) { log.warn("重复结算已被唯一索引拦截: tenantId={}, period={}", tenantId, period); }

异常处理

// ✅ 好:捕获具体异常,添加上下文 try { user = userRepository.findById(id); } catch (DataAccessException e) { throw new ServiceException("Failed to find user: " + id, e); }

// ✅ 好:资源自动关闭 try (InputStream is = new FileInputStream(file)) { // 使用资源 }

// ❌ 差:捕获过宽 catch (Exception e) { e.printStackTrace(); }

空值处理

// ✅ 使用 Optional public Optional<User> findById(Long id) { return userRepository.findById(id); }

// ✅ 参数校验 public void updateUser(User user) { Objects.requireNonNull(user, "user must not be null"); }

// ✅ 安全的空值处理 String name = Optional.ofNullable(user) .map(User::getName) .orElse("Unknown");

并发编程

// ✅ 使用 ExecutorService ExecutorService executor = Executors.newFixedThreadPool(10); Future<Result> future = executor.submit(() -> doWork());

// ✅ 使用 CompletableFuture CompletableFuture<User> future = CompletableFuture .supplyAsync(() -> findUser(id)) .thenApply(user -> enrichUser(user));

// ❌ 差:直接创建线程 new Thread(() -> doWork()).start();

测试规范 (JUnit 5)

class UserServiceTest { @Test @DisplayName("根据 ID 查找用户 - 用户存在时返回用户") void findById_whenUserExists_returnsUser() { // given when(userRepository.findById(1L)).thenReturn(Optional.of(expected));

    // when
    Optional&#x3C;User> result = userService.findById(1L);

    // then
    assertThat(result).isPresent();
    assertThat(result.get().getName()).isEqualTo("test");
}

}

Spring Boot 规范

// ✅ 构造函数注入 @Service @RequiredArgsConstructor public class UserService { private final UserRepository userRepository; private final EmailService emailService; }

// ✅ REST Controller @RestController @RequestMapping("/api/users") public class UserController { @GetMapping("/{id}") public ResponseEntity<UserDto> findById(@PathVariable Long id) { return userService.findById(id) .map(ResponseEntity::ok) .orElse(ResponseEntity.notFound().build()); } }

Auth Filter 降级原则

规则 说明

✅ optional-auth 路径遇到无效/过期/不完整 token 时降级为匿名访问 不应返回 401/403

❌ 禁止部分凭证用户体验差于匿名用户 如:临时 token 在公开接口返回 403

输入校验规范

规则 说明

❌ 禁止 @RequestBody 不加 @Valid

所有请求体必须校验

✅ DTO 字段加约束注解 @NotBlank 、@Size 、@Pattern 等

✅ 数值字段加范围约束 @Min 、@Max 、@Positive 等

✅ 分页参数加上限 size 必须 @Max(100) 防止大量查询

✅ 枚举/状态字段白名单校验 自定义校验器或 @Pattern

常见 DTO 字段校验速查:

字段类型 必须注解 说明

数量 quantity @NotNull @Min(1)

防止 0 或负数(负数可导致反向操作)

金额 amount/price @NotNull @Positive

或 @DecimalMin("0.01")

分页 size @Min(1) @Max(100)

防止 size=999999 拖垮数据库

分页 page @Min(1)

页码从 1 开始

百分比 rate @Min(0) @Max(100)

视业务定义范围

// ❌ 无校验,任意输入直接进入业务逻辑 @PostMapping("/ship") public Result ship(@RequestBody ShippingRequest request) { ... }

// ✅ 完整校验 @PostMapping("/ship") public Result ship(@RequestBody @Valid ShippingRequest request) { ... }

public record ShippingRequest( @NotNull Long orderId, @NotBlank @Size(max = 500) String shippingInfo, @Pattern(regexp = "pending|shipped|delivered") String giftStatus ) {}

// ❌ quantity 只有 @NotNull,负数会导致 Redis DECRBY 反向加库存 public record CreateOrderRequest( @NotNull Integer quantity // 可提交 0 或负数 ) {}

// ✅ 数量必须 >= 1 public record CreateOrderRequest( @NotNull @Min(1) Integer quantity ) {}

// ❌ 分页无上限,用户可传 size=999999 @GetMapping("/orders") public Result list(@RequestParam int page, @RequestParam int size) { ... }

// ✅ 分页参数加约束 @GetMapping("/orders") public Result list(@RequestParam @Min(1) int page, @RequestParam @Min(1) @Max(100) int size) { ... }

性能优化

陷阱 解决方案

N+1 查询 见「N+1 查询防范」章节

循环拼接字符串 使用 StringBuilder

频繁装箱拆箱 使用原始类型流

未指定集合初始容量 new ArrayList<>(size)

日志规范

// ✅ 参数化日志 log.debug("Finding user by id: {}", userId); log.info("User {} logged in successfully", username); log.error("Failed to process order {}", orderId, exception);

// ❌ 差:字符串拼接 log.debug("Finding user by id: " + userId);

详细参考

文件 内容

references/java-style.md

命名约定、异常处理、Spring Boot、测试规范

references/collections.md

不可变集合(Guava)、字符串分割

references/concurrency.md

线程池配置、CompletableFuture 超时

references/concurrency-db-patterns.md

Get-Or-Create 并发、N+1 防范、原子更新、Redis+DB 一致性

references/code-patterns.md

卫语句、枚举优化、策略工厂模式

references/date-time.md

日期加减、账期计算、禁止月末对齐

📋 本回复遵循:java-dev

  • [具体章节]

Source Transparency

This detail page is rendered from real SKILL.md content. Trust labels are metadata-based hints, not a safety guarantee.

Related Skills

Related by shared tags or category signals.

Coding

frontend-dev

No summary provided by upstream source.

Repository SourceNeeds Review
Coding

go-dev

No summary provided by upstream source.

Repository SourceNeeds Review
Coding

python-dev

No summary provided by upstream source.

Repository SourceNeeds Review