其他
Java:如何更优雅的处理空值?
点击上方 Java后端,选择 设为星标
优质文章,及时送达导语
业务中的空值
场景
public interface UserSearchService{
List<User> listUser();
User get(Integer id);
}
问题现场
listUser(): 查询用户列表 get(Integer id): 查询单个用户
listUser() 如果没有数据,那它是返回空集合还是null呢? get(Integer id) 如果没有这个对象,是抛异常还是返回null呢?
深入listUser研究
listUser()
public List<User> listUser(){
List<User> userList = userListRepostity.selectByExample(new UserExample());
if(CollectionUtils.isEmpty(userList)){//spring util工具类
return null;
}
return userList;
}
public List<User> listUser(){
List<User> userList = userListRepostity.selectByExample(new UserExample());
if(CollectionUtils.isEmpty(userList)){
return Lists.newArrayList();//guava类库提供的方式
}
return userList;
}
(List listUser())
,它一定会返回List,即使没有数据,它仍然会返回List(集合中没有任何元素);深入研究get方法
User get(Integer id)
public User get(Integer id){
return userRepository.selectByPrimaryKey(id);//从数据库中通过id直接获取实体对象
}
public interface UserSearchService{
/**
* 根据用户id获取用户信息
* @param id 用户id
* @return 用户实体
* @exception UserNotFoundException
*/
User get(Integer id);
}
public interface UserSearchService{
/**
* 根据用户id获取用户信息
* @param id 用户id
* @return 用户实体,此实体有可能是缺省值
*/
Optional<User> getOptional(Integer id);
}
public Optional<User> getOptional(Integer id){
return Optional.ofNullable(userRepository.selectByPrimaryKey(id));
}
深入入参
强制约束 文档性约束(弱提示)
public interface UserSearchService{
/**
* 根据用户id获取用户信息
* @param id 用户id
* @return 用户实体
* @exception UserNotFoundException
*/
User get(@NotNull Integer id);
/**
* 根据用户id获取用户信息
* @param id 用户id
* @return 用户实体,此实体有可能是缺省值
*/
Optional<User> getOptional(@NotNull Integer id);
}
public interface UserSearchService{
/**
* 根据用户id获取用户信息
* @param id 用户id
* @return 用户实体
* @exception UserNotFoundException
*/
User get(@NonNull Integer id);
/**
* 根据用户id获取用户信息
* @param id 用户id
* @return 用户实体,此实体有可能是缺省值
*/
Optional<User> getOptional(@NonNull Integer id);
}
小结
空集合返回值 :如果有集合这样返回值时,除非真的有说服自己的理由,否则,一定要返回空集合,而不是null Optional: 如果你的代码是jdk8,就引入它!如果不是,则使用Guava的Optional,或者升级jdk版本!它很大程度的能增加了接口的可读性! jsr 303: 如果新的项目正在开发,不防加上这个试试!一定有一种特别爽的感觉! jsr 305: 如果老的项目在你的手上,你可以尝试的加上这种文档型注解,有助于你后期的重构,或者新功能增加了,对于老接口的理解!
空对象模式
场景
static class PersonDTO{
private String dtoName;
private String dtoAge;
}
static class Person{
private String name;
private String age;
}
@Test
public void shouldConvertDTO(){
PersonDTO personDTO = new PersonDTO();
Person person = new Person();
if(!Objects.isNull(person)){
personDTO.setDtoAge(person.getAge());
personDTO.setDtoName(person.getName());
}else{
personDTO.setDtoAge("");
personDTO.setDtoName("");
}
}
优化修改
static class NullPerson extends Person{
@Override
public String getAge() {
return "";
}
@Override
public String getName() {
return "";
}
}
@Test
public void shouldConvertDTO(){
PersonDTO personDTO = new PersonDTO();
Person person = getPerson();
personDTO.setDtoAge(person.getAge());
personDTO.setDtoName(person.getName());
}
private Person getPerson(){
return new NullPerson();//如果Person是null ,则返回空对象
}
使用Optional可以进行优化
@Test
public void shouldConvertDTO(){
PersonDTO personDTO = new PersonDTO();
Optional.ofNullable(getPerson()).ifPresent(person -> {
personDTO.setDtoAge(person.getAge());
personDTO.setDtoName(person.getName());
});
}
private Person getPerson(){
return null;
}
Optioanl的正确使用
Optional不要作为参数
public interface UserService{
List<User> listUser(Optional<String> username);
}
“如果username是absent,是返回空集合吗?还是返回全部的用户数据集合?”
public interface UserService{
List<User> listUser(String username);
List<User> listUser();
}
Optional作为返回值
当个实体的返回
public interface UserService{
Optional<User> get(Integer id);
}
集合实体的返回
public interface UserService{
Optional<List<User>> listUser();
}
使用Optional变量
Optional<User> userOpt = ...
一定不能直接使用get ,如果这样用,就丧失了Optional本身的含义 ( 比如userOp.get() ) 不要直接使用getOrThrow ,如果你有这样的需求:获取不到就抛异常。那就要考虑,是否是调用的接口设计的是否合理
getter中的使用
小结
当使用值为空的情况,并非源于错误时,可以使用Optional! Optional不要用于集合操作! 不要滥用Optional,比如在java bean的getter中!
【END】