前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Java优化_解决if嵌套过多

Java优化_解决if嵌套过多

作者头像
全栈程序员站长
发布2022-09-29 10:37:14
2K0
发布2022-09-29 10:37:14
举报
文章被收录于专栏:全栈程序员必看

大家好,又见面了,我是你们的朋友全栈

文章目录


一、使用策略模式

彻底消灭if-else嵌套 设计模式-策略模式 如何干掉 Spring Boot 中大片的 if else? 业务代码中, if else 使用策略模式优化 设计模式 策略模式 极简策略模式 (妙用枚举) 责任链模式+策略模式

二、其他方案

1.分解条件表达式

代码语言:javascript
复制
if (date.before (SUMMER_START) || date.after(SUMMER_END)) { 
   
    charge = quantity * _winterRate + _winterServiceCharge;
} else { 
   
    charge = quantity * _summerRate
}

这种代码很多人可能都觉得没必要去提取方法,但是如果我们想要看懂这段代码,还是必须的去想想才知道在做什么;接下来我们修改一下

代码语言:javascript
复制
if (notSummer(date)) { 

charge = winterCharge(quantity);
} else { 

charge = summerCharge(quantity);
}
private boolean notSummer(Date date){ 

date.before (SUMMER_START) || date.after(SUMMER_END)
}
private double summerCharge(int quantity) { 

return quantity * _summerRate;
}
private double winterCharge(int quantity) { 

return quantity * _winterRate + _winterServiceCharge;
}

2.合并重复的条件判断

代码语言:javascript
复制
double disabilityAmount () { 

if(_seniortiy <2 ) 
return 0;
if(_monthsDisabled > 12)
return 0;
if(_isPartTime)
return 0;
// 省略...
}

这里的条件返回的结果都是一样的,那么我们先把条件合并起来

代码语言:javascript
复制
double disabilityAmount () { 

if(_seniortiy <2 || _monthsDisabled > 12 || _isPartTime) { 

return 0;
}
// 省略...
}

接下来我们再来把判断条件判断条件抽取成方法提高可读性

代码语言:javascript
复制
double disabilityAmount () { 

if(isNotEligibleForDisableility()) { 

return 0;
}
// 省略...
}
boolean isNotEligibleForDisableility() { 

return _seniortiy <2 || _monthsDisabled > 12 || _isPartTime;
}

举例2:

代码语言:javascript
复制
if(onVacation()) { 

if(lengthOfService() > 10) { 

return 2;
}
}
return 1;

合并之后的代码

代码语言:javascript
复制
if(onVacation() && lengthOfService() > 10){ 

return 2
}
return 1;

接着我们可以使用三元操作符更加简化,修改后的代码:

代码语言:javascript
复制
return onVacation() && lengthOfService() > 10 ? 2 : 1;

3.提前判断返回

如下语句

代码语言:javascript
复制
if(condition){ 

//dost
}else{ 

return ;
}

改为

代码语言:javascript
复制
if(!condition){ 

return ;
}
//dost

避免一些不必要的分支,让代码更精炼。

4.引入断言工具类

比如下面这段代码:

代码语言:javascript
复制
public void getProjectLimit(String project){ 

if(project == null){ 

throw new RuntimeException("project can not null");
}
doSomething();
}

加入Spring的断言后的代码 或者自定义断言 Assert.java

代码语言:javascript
复制
public void getProjectLimit(String project){ 

Assert.notNull(project,"project can not null");
doSomething();
}

5.善用 Optional

在项目中,总少不了一些非空的判断,可能大部分人还是如下的用法

代码语言:javascript
复制
if(null == user){ 

//action1
}else{ 

//action2
}

这时候该掏出Optional这个秘密武器了,它可以让非空校验更加优雅,间接的减少if操作。没了解过Optional的同学可自行Google,这里就不再赘述。

代码语言:javascript
复制
// 如果dtolgetProductType()为空 则默认为0
Integer productType = Optional.ofNullable(dto.getProductType()).orElse(0)
user = Optional.ofNullable(user).orElse(new User());
Optional<User> userOptional = Optional.ofNullable(user);
userOptional.map(action1).orElse(action2);

上面的代码跟第一段是等效的,通过一些新特性让代码更加紧凑。

6.使用枚举

代码语言:javascript
复制
public enum Status { 

NEW(0),RUNNABLE(1),RUNNING(2),BLOCKED(3),DEAD(4);
public int statusCode;
Status(int statusCode){ 

this.statusCode = statusCode;
}
}

那么我们在使用的时候就可以直接通过枚举调用了。 int statusCode = Status.valueOf(“NEW”).statusCode; 优雅的解决了下面代码赋值的方式

代码语言:javascript
复制
if(param.equals("NEW")){ 

statusCode = 0;
}else if(param.equals("RUNNABLE")){ 

statusCode = 1;
}

使用枚举优化if else实现2

7.枚举多态

代码语言:javascript
复制
int attackPower(Attacker attacker) { 

return AttackerType.valueOf(attacker.getType()).getAttackPower();
}
enum AttackerType { 

Bartizan("箭塔") { 

@Override
public int getAttackPower() { 

return 100;
}
},
Archer("弓箭手") { 

@Override
public int getAttackPower() { 

return 50;
}
},
Tank("坦克") { 

@Override
public int getAttackPower() { 

return 800;
}
};
private String label;
Attacker(String label) { 

this.label = label;
}
public String getLabel() { 

return label;
}
public int getAttackPower() { 

throw new RuntimeException("Can not support the method");
}
}

8.类多态

if else 示例1:

代码语言:javascript
复制
  String medalType = "guest";
if ("guest".equals(medalType)) { 

System.out.println("嘉宾勋章");
} else if ("vip".equals(medalType)) { 

System.out.println("会员勋章");
} else if ("guard".equals(medalType)) { 

System.out.println("守护勋章");
}

多态优化:

代码语言:javascript
复制
//勋章接口
public interface IMedalService { 

void showMedal();
}
//守护勋章策略实现类
public class GuardMedalServiceImpl implements IMedalService { 

@Override
public void showMedal() { 

System.out.println("展示守护勋章");
}
}
//嘉宾勋章策略实现类
public class GuestMedalServiceImpl implements IMedalService { 

@Override
public void showMedal() { 

System.out.println("嘉宾勋章");
}
}
//勋章服务工厂类
public class MedalServicesFactory { 

private static final Map<String, IMedalService> map = new HashMap<>();
static { 

map.put("guard", new GuardMedalServiceImpl());
map.put("vip", new VipMedalServiceImpl());
map.put("guest", new GuestMedalServiceImpl());
}
public static IMedalService getMedalService(String medalType) { 

return map.get(medalType);
}
}

示例2:

代码语言:javascript
复制
int attackPower(Attacker attacker) { 

return attacker.getAttackPower();
}
interface Attacker { 

default int getAttackPower() { 

throw new RuntimeException("Can not support the method");
}
}
class Bartizan implements Attacker { 

public int getAttackPower() { 

return 100 * getLevel();
}
}
class Archer implements Attacker { 

public int getAttackPower() { 

return 50 * getLevel();
}
}
class Tank implements Attacker { 

public int getAttackPower() { 

return 800 * getLevel();
}
}

9.表驱动法

来自Google的解释:表驱动法是一种编程模式,它的本质是,从表里查询信息来代替逻辑语句(if,case)。下面看一个案例,通过月份来获取当月的天数(仅作为案例演示,获取2月份的数据不严谨),普通做法:

代码语言:javascript
复制
int getMonthDays(int month){ 

switch(month){ 

case 1:return 31;break;
case 2:return 29;break;
case 3:return 31;break;
case 4:return 30;break;
case 5:return 31;break;
case 6:return 30;break;
case 7:return 31;break;
case 8:return 31;break;
case 9:return 30;break;
case 10:return 31;break;
case 11:return 30;break;
case 12:return 31;break;
default:return 0;
}
}

表驱动法实现方式

代码语言:javascript
复制
int monthDays[12] = { 
31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};
int getMonthDays(int month){ 

return monthDays[--month];
}

其实这里的表就是数组而已,通过直接查询数组来获得需要的数据,那么同理,Map之类的容器也可以成为我们编程概念中的表。

代码语言:javascript
复制
Map<?, Function<?> action> actionsMap = new HashMap<>();
// 初试配置对应动作
actionsMap.put(value1, (someParams) -> { 
 doAction1(someParams)});
actionsMap.put(value2, (someParams) -> { 
 doAction2(someParams)});
actionsMap.put(value3, (someParams) -> { 
 doAction3(someParams)});
// 省略 null 判断
actionsMap.get(param).apply(someParams);

10. Map + Java8 函数式接口

通过Java8的lambda表达式,我们把需要执行东西存进value中,调用的时候通过匹配key的方式进行。

代码语言:javascript
复制
@Service
public class QueryGrantTypeService { 

@Autowired
private GrantTypeSerive grantTypeSerive;
private final Map<String, Function<String, String>> grantTypeMap = new HashMap<>();
/** * 初始化业务分派逻辑,代替了if-else部分 * key: 优惠券类型 * value: lambda表达式,最终会获得该优惠券的发放方式 */
@PostConstruct
public void dispatcherInit() { 

grantTypeMap.put("红包", resourceId -> grantTypeSerive.redPaper(resourceId));
grantTypeMap.put("购物券", resourceId -> grantTypeSerive.shopping(resourceId));
grantTypeMap.put("vip会员", resourceId -> grantTypeSerive.vip(resourceId));
}
public String getResult(String resourceType, String resourceId) { 

// Controller根据 优惠券类型resourceType、编码resourceId 去查询 发放方式grantType
Function<String, String> result = grantTypeMap.get(resourceType);
if (result != null) { 

// 传入 resourceId 执行这段表达式获得String型的grantType
return result.apply(resourceId);
}
return "查询不到该优惠券的发放方式";
}
}
@Service
public class GrantTypeSerive { 

public String redPaper(String resourceId) { 

//红包的发放方式
return "每周末9点发放";
}
public String shopping(String resourceId) { 

//购物券的发放方式
return "每周三9点发放";
}
public String vip(String resourceId) { 

//qq会员的发放方式
return "每周一0点开始秒杀";
}
}

调用:

代码语言:javascript
复制
@RestController
public class GrantTypeController { 

@Autowired
private QueryGrantTypeService queryGrantTypeService;
@PostMapping("/grantType")
public String test(String resourceName){ 

return queryGrantTypeService.getResult(resourceName);
}
}

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/193646.html原文链接:https://javaforall.cn

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2022年9月15日 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 文章目录
  • 一、使用策略模式
  • 二、其他方案
    • 1.分解条件表达式
      • 2.合并重复的条件判断
        • 3.提前判断返回
          • 4.引入断言工具类
            • 5.善用 Optional
              • 6.使用枚举
                • 7.枚举多态
                  • 8.类多态
                    • 9.表驱动法
                      • 10. Map + Java8 函数式接口
                      相关产品与服务
                      容器服务
                      腾讯云容器服务(Tencent Kubernetes Engine, TKE)基于原生 kubernetes 提供以容器为核心的、高度可扩展的高性能容器管理服务,覆盖 Serverless、边缘计算、分布式云等多种业务部署场景,业内首创单个集群兼容多种计算节点的容器资源管理模式。同时产品作为云原生 Finops 领先布道者,主导开源项目Crane,全面助力客户实现资源优化、成本控制。
                      领券
                      问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档