本文将引导你完成一系列从传统的命令式代码重构到Java 8函数代码,要从本文中获得最大收益,你应该具备Java 8函数的一些实践经验。
1)优先于匿名Lambda的命名函数
为了热身,让我们从简单的任务开始,将一些用户的详细信息带到UI。我们将从实体列表的开始,将User 转换到 UserDto:
public List<UserDto> getAllUsers() { List<User> users = userRepo.findAll(); List<UserDto> dtos = new ArrayList<>(); for (User user : users) { UserDto dto = new UserDto(); dto.setUsername(user.getUsername()); dto.setFullName(user.getFirstName() + " " + user.getLastName().toUpperCase()); dto.setActive(user.getDeactivationDate() == null); dtos.add(dto); } return dtos; }
|
但是,我对这段代码并不感到自豪,因为我很可能会为许多用例重复编写类似的代码。那么,让我们使用Java 8:
public List<UserDto> getAllUsers() { return userRepo.findAll().stream() .map(user -> { UserDto dto = new UserDto(); dto.setUsername(user.getUsername()); dto.setFullName(user.getFirstName() + " " + user.getLastName().toUpperCase()); dto.setActive(user.getDeactivationDate() == null); return dto; }) .collect(toList());
|
虽然不错,但是,我仍然不满意。我写的这个lambda演示了一个“匿名函数”。作为一个干净的代码疯子,我有一个问题 - 我想要富有表现力的名字。所以,我很快将lambda内容提取到一个单独的方法中:
public List<UserDto> getAllUsers() { return userRepo.findAll().stream().map(this::toDto).collect(toList()); } private UserDto toDto(User user) { UserDto dto = new UserDto(); dto.setUsername(user.getUsername()); dto.setFullName(user.getFirstName() + " " + user.getLastName().toUpperCase()); dto.setActive(user.getDeactivationDate() == null); return dto; }
|
代码比在之前的版本中更简单,但现在它稍微好一些。
这种User到DTO的逻辑转换可以直接放在DTO构造函数中:public class UserFacade { private UserRepo userRepo; public List<UserDto> getAllUsers() { return userRepo.findAll().stream().map(UserDto::new).collect(toList()); } } public class UserDto { private String username; private String fullName; private boolean active; public UserDto(User user) { username = user.getUsername(); fullName = user.getFirstName() + " " + user.getLastName().toUpperCase(); active = user.getDeactivationDate() == null; } ... }
|
现在,让我们假设这个转换需要一些其他组件的帮助,我们希望使用Spring,Guice,CDI等注入。但是,在我们实例化的类中注入依赖项需要非常复杂的代码。如果这个转换过于复杂,我们应该将它移到一个单独的UserMapper类并从那里引用它:
@Service public class UserFacade { @Autowired private UserRepo userRepo; @Autowired private UserMapper mapper; public List<UserDto> getAllUsers() { return userRepo.findAll().stream().map(mapper::toDto).collect(toList()); } } @Component public class UserMapper { @Autowired private OtherClass otherClass; public UserDto toDto(User user) { UserDto dto = new UserDto(); dto.setUsername(user.getUsername()); ... // code using otherClass return dto; } }
|
关键点是:始终将复杂的lambda提取到具有表达名称的函数中,然后可以使用以下四点(::)来引用:
- 如在同一个类,使用this::;
- 在另外一个类似于(mapper::);
- 一些静态助手方法(SomeClass::);
- Stream 中条目类型(Item::);
- 甚至一些构造函数(),如果它足够简单;UserDto::new
记住,不要使用匿名类型。在这个GitHub存储库中提交了练习的每个阶段,所以请随意浏览存储库以查看所有内容。