java if else 过多_如何解决代码中过多的 if else ?

ef33752dfcd66502d393ba5503c56b8f.png

先来一张镇楼图感受一下 if else 的魔法吧。

167616ef98e542e8dd83f64b467c04a6.png

一、由一个几百行 if 引发的思考

有个场景,50张字典表,需要为其他服务提供一个统一的接口来校验用户输入的字典表 id 是否合法。

校验逻辑已经很清晰了,根据参数选择对应的表校验 id 是否存在。if("table_a".equals(table)) {

// check id

}

if("table_b".equals(table)) {

// check id

}

if("table_c".equals(table)) {

// check id

}...

再加上参数校验,函数调用,@Autowired bean 等等,一坨几百行的代码 ok 了。再新加表再加 if else 就行了,😋 完美。

如此,N 年后另一个可怜的小伙伴就看到这坨东西。

二、KO 这些 if else

回想上面的场景,实际上就是要根据表名去确定 id 是否存在表中,那么只要将表名与操作对应起来就行了。故而采用哈希表的形式,将表名与操作对应起来。部分代码如下:// 用于保存表与 Function 的对应关系

private final Map> actionMappings = new ConcurrentHashMap<>(50);

@PostConstruct

private void init() {

// map 初始化

actionMappings.put(TableConstants.TABLE_A, (params) -> tableAManager.getById(params));

}

/**

* 校验逻辑

*

*@param table

*@param id

*/

public boolean valid(String table, Long id) {

Object object = actionMappings.get(table).apply(id);

// 不存在则校验失败

return !Objects.isNull(object);

}

如此,N 多行 if 被消除了,这种编程方式也叫做表驱动。虽然 if 没有了,但是在初始化 actionMappings 的时候还是很多行重复代码。下面采用注解方式解决:/**

* 标记此注解的 bean 会加入基础数据校验全局 Function Map

*

* @author aysaml

* @date 2020/5/7

*/

@Documented

@Inherited

@Target(ElementType.TYPE)

@Retention(RetentionPolicy.RUNTIME)

public @interface ValidHandler {

TABLE_ENUM value();

}

value 是表名枚举,在需要的类上面加上此注解即可。同时定义一个 context 用来专门存储 actionMappings 。/**

* 数据校验上下文对象,用于保存各表的 Function Map

*

* @author aysaml

* @date 2020/5/7

*/

@Component

public class CommonDataValidContext {

private static final Logger LOGGER = LoggerFactory.getLogger(CommonDataValidContext.class);

private final Map> actionMappings = new ConcurrentHashMap<>(50);

/**

* 方法加入 mappings

*

* @param model 表名

* @param action 方法

*/

public void putAction(String model, Function action) {

if (!Objects.isNull(action)) {

actionMappings.put(model, action);

LOGGER.info(

"[{}] add to CommonDataValidContext actionMappings, actionMappings size : {}",

model,

actionMappings.size());

}

}

/**

* 执行方法获取返回结果

*

* @param model

* @param param

* @return

*/

public

R apply(String model, P param) {

if (actionMappings.containsKey(model)) {

return (R) actionMappings.get(model).apply(param);

} else {

LOGGER.error("执行数据校验时model={}不存在!", model);

throw new RuntimeException("基础数据校验时发生错误:" + model + "表不存在!");

}

}

/**

* 判断 mappings 中是否含有给定 model 的处理方法

*

* @param model

* @return

*/

public boolean containsKey(String model) {

return actionMappings.containsKey(model);

}

/**

* 校验执行方法的返回值是否为空

*

* @param model

* @param param

* @param

* @return

*/

public

boolean validResultIsNull(String model, P param) {

return Objects.isNull(this.apply(model, param));

}

}

然后通过监听器的方式,将含有 ValidHandler 注解的方法加入 actionMappings 。/**

* 基础数据校验处理方法监听器

*

* @author aysaml

* @date 2020/5/7

*/

@Component

public class CommonValidActionListener implements ApplicationListener {

@Override

public void onApplicationEvent(ContextRefreshedEvent event) {

Map beans =

event.getApplicationContext().getBeansWithAnnotation(ValidHandler.class);

CommonDataValidContext commonDataValidContext =

event.getApplicationContext().getBean(CommonDataValidContext.class);

beans.forEach(

(name, bean) -> {

ValidHandler validHandler = bean.getClass().getAnnotation(ValidHandler.class);

commonDataValidContext.putAction(

validHandler.value().code(),

(param) -> {

try {

return bean.getClass().getMethod("getById", Long.class).invoke(bean, param);

} catch (Exception e) {

e.printStackTrace();

}

return null;

});

});

}

}

三、更多消除 if else 的方法。

1. 提前return

这样可以使代码在逻辑表达上会更清晰,如下:if (condition) {

// do something

} else {

return xxx;

}

按照逆向思维来,优化如下:if (!condition) {

return xxx;

}

// do something

还有一种常见的傻瓜编程(如有冒犯,敬请见谅,对码不对人🙏 ):if(a > 0) {

return true;

} else {

return false;

}

话不多说了,直接 return a > 0; 不香吗?

2. 策略模式

简单来说就是根据不同的参数执行不同的业务逻辑。

如下:if (status == 0) {

// 业务逻辑处理 0

} else if (status == 1) {

// 业务逻辑处理 1

} else if (status == 2) {

// 业务逻辑处理 2

} else if (status == 3) {

// 业务逻辑处理 3

}...

优化如下:多态interface A {

void run() throws Exception;

}

class A0 implements A {

@Override

void run() throws Exception {

// 业务逻辑处理 0

}

}

class A1 implements A {

@Override

void run() throws Exception {

// 业务逻辑处理 1

}

}

// ...

然后策略对象存放在一个 Map 中,如下:A a = map.get(param);

a.run();

2.2 枚举public enum Status {

NEW(0) {

@Override

void run() {

//do something

}

},

RUNNABLE(1) {

@Override

void run() {

//do something

}

};

public int statusCode;

abstract void run();

Status(int statusCode){

this.statusCode = statusCode;

}

}

重新定义策略枚举public enum Aenum {

A_0 {

@Override

void run() {

//do something

}

},

A_1 {

@Override

void run() {

//do something

}

};

//...

abstract void run();

}

通过枚举优化之后的代码如下Aenum a = Aenum.valueOf(param);

a.run();

3. Java 8 的 Optional

Optional主要用于非空判断,是 Java 8 提供的新特性。

使用之前:if (user == null) {

//do action 1

} else {

//do action2

}

如果登录用户为空,执行action1,否则执行action 2,使用Optional优化之后,让非空校验更加优雅,间接的减少if操作Optional userOptional = Optional.ofNullable(user);

userOptional.map(action1).orElse(action2);

4. 决策表

就是上面的表驱动编程方法。欢迎访问个人博客 获取更多知识分享。

b739ec46bb5c46d9c0aa4ce35ba1ea56.png

关于找一找教程网

本站文章仅代表作者观点,不代表本站立场,所有文章非营利性免费分享。

本站提供了软件编程、网站开发技术、服务器运维、人工智能等等IT技术文章,希望广大程序员努力学习,让我们用科技改变世界。

[如何解决代码中过多的 if else ?]http://www.zyiz.net/tech/detail-146122.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是几种Java优化if-else代码的方法: 1. 使用策略枚举来优化if-else。策略枚举是一种常用的设计模式,它可以将一组算法封装起来,并使它们之间可以互换。这种方法可以避免使用大量的if-else语句,提高代码的可读性和可维护性。 2. 使用switch语句来代替if-else。在某些情况下,使用switch语句可以使代码更加简洁和易于理解。但是,如果case语句过多,也会导致代码变得冗长和难以维护。 3. 使用多态来代替if-else。多态是面向对象编程的一个重要概念,它可以使代码更加灵活和可扩展。通过将不同的行为封装在不同的类,可以避免使用大量的if-else语句。 4. 使用Optional类优化if-else。Optional类是Java 8引入的一个新特性,它可以使代码更加简洁和易于理解。通过使用Optional类,可以避免使用大量的if-else语句来处理空值的情况。 以下是一个使用策略枚举来优化if-else的例子: ```java public enum Discount { VIP(0.8), NORMAL(1.0), NEW(0.9); private double discount; Discount(double discount) { this.discount = discount; } public double getDiscount() { return discount; } } public class Order { private double price; private Discount discount; public Order(double price, Discount discount) { this.price = price; this.discount = discount; } public double calculatePrice() { return price * discount.getDiscount(); } } // 使用策略枚举来计算订单价格 Order order = new Order(100, Discount.VIP); double price = order.calculatePrice(); System.out.println(price); // 输出:80.0 ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值