Java如何优化大量的if else |
您所在的位置:网站首页 › ifelse语句可以处理多个 › Java如何优化大量的if else |
文章目录
一、使用策略模式二、其他方案1.分解条件表达式2.合并重复的条件判断3.提前判断返回4.引入断言工具类5.善用 Optional6.使用枚举7.枚举多态8.类多态9.表驱动法10. Map + Java8 函数式接口11. Map + Spring Bean
一、使用策略模式
彻底消灭if-else嵌套 设计模式-策略模式 如何干掉 Spring Boot 中大片的 if else? 业务代码中, if else 使用策略模式优化 设计模式 策略模式 极简策略模式 (妙用枚举) 责任链模式+策略模式 二、其他方案 1.分解条件表达式 if (date.before (SUMMER_START) || date.after(SUMMER_END)) { charge = quantity * _winterRate + _winterServiceCharge; } else { charge = quantity * _summerRate }这种代码很多人可能都觉得没必要去提取方法,但是如果我们想要看懂这段代码,还是必须的去想想才知道在做什么;接下来我们修改一下 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.合并重复的条件判断 double disabilityAmount () { if(_seniortiy 12) return 0; if(_isPartTime) return 0; // 省略... }这里的条件返回的结果都是一样的,那么我们先把条件合并起来 double disabilityAmount () { if(_seniortiy 12 || _isPartTime) { return 0; } // 省略... }接下来我们再来把判断条件判断条件抽取成方法提高可读性 double disabilityAmount () { if(isNotEligibleForDisableility()) { return 0; } // 省略... } boolean isNotEligibleForDisableility() { return _seniortiy 12 || _isPartTime; }举例2: if(onVacation()) { if(lengthOfService() > 10) { return 2; } } return 1;合并之后的代码 if(onVacation() && lengthOfService() > 10){ return 2 } return 1;接着我们可以使用三元操作符更加简化,修改后的代码: return onVacation() && lengthOfService() > 10 ? 2 : 1; 3.提前判断返回如下语句 if(condition){ //dost }else{ return ; }改为 if(!condition){ return ; } //dost避免一些不必要的分支,让代码更精炼。 4.引入断言工具类比如下面这段代码: public void getProjectLimit(String project){ if(project == null){ throw new RuntimeException("project can not null"); } doSomething(); }加入Spring的断言后的代码 或者自定义断言 Assert.java public void getProjectLimit(String project){ Assert.notNull(project,"project can not null"); doSomething(); } 5.善用 Optional在项目中,总少不了一些非空的判断,可能大部分人还是如下的用法 if(null == user){ //action1 }else{ //action2 }这时候该掏出Optional这个秘密武器了,它可以让非空校验更加优雅,间接的减少if操作。没了解过Optional的同学可自行Google,这里就不再赘述。 // 如果dtolgetProductType()为空 则默认为0 Integer productType = Optional.ofNullable(dto.getProductType()).orElse(0) user = Optional.ofNullable(user).orElse(new User()); Optional userOptional = Optional.ofNullable(user); userOptional.map(action1).orElse(action2);上面的代码跟第一段是等效的,通过一些新特性让代码更加紧凑。 6.使用枚举 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; 优雅的解决了下面代码赋值的方式 if(param.equals("NEW")){ statusCode = 0; }else if(param.equals("RUNNABLE")){ statusCode = 1; }使用枚举优化if else实现2 7.枚举多态 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: 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("守护勋章"); }多态优化: //勋章接口 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 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: 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月份的数据不严谨),普通做法: 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; } }表驱动法实现方式 int monthDays[12] = {31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}; int getMonthDays(int month){ return monthDays[--month]; }其实这里的表就是数组而已,通过直接查询数组来获得需要的数据,那么同理,Map之类的容器也可以成为我们编程概念中的表。 Map 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的方式进行。 @Service public class QueryGrantTypeService { @Autowired private GrantTypeSerive grantTypeSerive; private final Map 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 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点开始秒杀"; } }调用: @RestController public class GrantTypeController { @Autowired private QueryGrantTypeService queryGrantTypeService; @PostMapping("/grantType") public String test(String resourceName){ return queryGrantTypeService.getResult(resourceName); } } 11. Map + Spring Bean多租户代码重构优化 OrderFactory工厂 public class OrderFactory implements ApplicationContextAware { // 租户A private static final String ORDER_TENANT_A = "A"; // 租户B private static final String ORDER_TENANT_B = "B"; @Autowired private OrderTenantConfig config; private static Map orderActionMap = new HashMap(2); /** * 根据租户id获取对应租户 **/ public static OrderAction getOrderAction(String tenantId) { OrderAction orderAction = orderActionMap.get(tenantId); if (ObjectUtil.isEmpty(orderAction)){ throw new OrderBusinessException(OrderCodeEnum.TENANT_ID_IS_NULL); } return orderAction; } @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { // AOrderAction、BOrderAction 两个租户各自的实现类 orderActionMap.put(config.getMap().get(ORDER_TENANT_A), applicationContext.getBean(AOrderAction.class)); orderActionMap.put(config.getMap().get(ORDER_TENANT_B), applicationContext.getBean(BOrderAction.class)); } } Controller调用: @PostMapping("/infoByNamePage") @ApiOperation(value = "分页查询", notes = "分页查询") public R infoByNamePage(@Valid @RequestBody OrderInfoVoRequest orderInfoVoRequest) { OrderAction orderAction = OrderFactory.getOrderAction(CurrentUserUtil.getTenantId()); return orderAction.infoByNamePage(orderInfoVoRequest); } |
今日新闻 |
点击排行 |
|
推荐新闻 |
图片新闻 |
|
专题文章 |
CopyRight 2018-2019 实验室设备网 版权所有 win10的实时保护怎么永久关闭 |