Java优化_解决if嵌套过多

Java优化_解决if嵌套过多文章目录 一、使用策略模式 二、其他方案 1.分解条件表达式 2.合并重复的条件判断 3.提前判断返回 4.引入断言工具类 5.善用 Optional 6.使用枚举 7.枚举多态 8.类多态 9

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺


一、使用策略模式

彻底消灭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 <2 ) 
        return 0;
    if(_monthsDisabled > 12)
        return 0;
    if(_isPartTime)
        return 0;
    // 省略...
}

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

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

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

double disabilityAmount () { 
   
    if(isNotEligibleForDisableility()) { 
   
        return 0;
    }
    // 省略...
}

boolean isNotEligibleForDisableility() { 
   
    return _seniortiy <2 || _monthsDisabled > 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<User> 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<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:

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;
		defaultreturn 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<?, 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的方式进行。

@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点开始秒杀";
    }
}

调用:

@RestController
public class GrantTypeController { 
   

    @Autowired
    private QueryGrantTypeService queryGrantTypeService;

    @PostMapping("/grantType")
    public String test(String resourceName){ 
   
        return queryGrantTypeService.getResult(resourceName);
    }
}
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

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

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • c(多线程编程)

    c(多线程编程)c 多线程编程什么是多线程 相必有些程序或者计算机基础的就会有所了解 我就不做过多赘述了 确实不知道的 可以反手百度 线程开启方法之 委托线程开启 c 中开启多线程的方法之一 是调用委托的 BeginInvoke 方法 可以当即为该委托打开一个新的线程来跑其包含的方法 参数 BeginInvoke 方法有俩个默认的参数 如果当前委托是有参数的 那么委托的参数写在该方法默认参数的前面 判断线程状态 BeginInvoke 方法有个 IAsyncResult 类型的返回值 调用该返回值的 Is

    2025年12月9日
    6
  • PotPlayer 高逼格无边框的本地播放器

    PotPlayer 高逼格无边框的本地播放器PotPlayer安装以及设置可以在官网下载安装包:http://potplayer.daum.net然后安装……运行……嗯,是的,PotPlayer的官方皮肤界面就是这么样貌平平,一点都不逼格。其实,如果你之前还用更早的版本,大概现在的默认界面已经可以说是貌美如花了(笑),下面我来教大家如何把播放器设置成无边框的。1右键→选项…→进入设置界面。(或者直接按F5)…

    2022年5月21日
    141
  • Nginx修改默认端口80

    Nginx修改默认端口80前言    安装流程请参考我的文章–Windows下安装Nginx。    博客地址:https://blog.csdn.net/zengwende/article/details/86610692修改步骤1、打开Nginx的配置文件nginx.conf2、修改默认端口的值即可(nginx默认的端口为80) …

    2025年10月6日
    3
  • 网约技师APP详细设计说明书「建议收藏」

    网约技师APP详细设计说明书「建议收藏」目录1引言31.1编写目的31.2背景31.3定义31.4参考资料42程序系统的结构43登录程序Login()设计说明53.1程序描述53.2功能63.3性能63.4输人项63.5输出项73.6算法73.7流程逻辑73.8接口83.9存储分配83.10注释设计

    2022年6月4日
    34
  • Java中重载与重写的特点和区别[通俗易懂]

    Java中重载与重写的特点和区别[通俗易懂]一、重载(overload)重载:发生在Java的同一个类里面,通过定义相同的方法,定义不同的参数列表来区分开这些相同的方法,参数列表的不同可以是参数的类型、参数的个数和参数的顺序。当我们调用这些方法的时候通过传入参数的不同,在运行时Java虚拟机会自行调用不同的方法来进行处理。由于Java的修饰符都可以访问同一个类里面的方法,所以方法的重载与方法的修饰符无关;Java的返回值不能区别这些方法,因为返回值是在运行后才体现出来的,而两个参数列表相同仅仅是返回值相同的方法在Java编译时就会报错,虚拟机在运行

    2022年7月8日
    25
  • java编译原理

    java编译原理4.Java编译原理1.javac是什么?(1)javac是一种编译器,能够将一种语言规范转换成另一种用语言规范,通常编译器是将便于人们理解的语言规范成机器容易理解的语言规范。(2)javac的任务就是将java源代码语言转换成jvm能够识别的语言,然后jvm将jvm语言再转化成当前机器能够识别的语言(这样使得对开发者屏蔽与机器相关的细节,并且使得语言的执行与平台无关)2.javac编译器的基本结…

    2022年5月9日
    39

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号