Java 8函数式编程模式:不要使用匿名函数


本文将引导你完成一系列从传统的命令式代码重构到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存储库中提交了练习的每个阶段,所以请随意浏览存储库以查看所有内容。