聊聊java中一些减少if-else 的编码方式!

本文分享了四种实用的编码技巧,旨在减少if-else语句的使用,提高代码的可读性和维护性。包括提前return、使用三目运算符、Assert断言及Optional对象,这些方法有助于简化复杂的条件判断逻辑。

01 前言

前段时间在阅读别人所写的代码的时候 , 发现其中一些业务相关的方法体内 , 出现了比较多的if-else语句多层嵌套的情况 . 首先我个人不是不提倡写if-else语句 , 不得不说 , 很多时候 , 在写某些逻辑 使用if-else 去做判断 , 代码看起来还是十分直观的 , 但是如果滥用if-else , 形成多层嵌套或者形成, 其中每个case 还包含了大量的逻辑 , 此时从可读性来说 , 使用if-else就有点得不偿失了 . 而且某些时候 , 可能并不需这么多的if-else , 或者是可以使用其他编码方式从而达到减少的if-else 的效果 .

减少if-else 的使用的方式有很多 , 例如设计模式层面的策略模式或者是责任链模式 . 而这里跟大家分享一下一些个人在日常编码过程中经常用到的 , 比较简单的 、从编码习惯层面上的方式 , 去一些减少不必要if-else使用 . 由于本人只是一个小菜鸟 , 如果有写得不对的地方 , 恳请批评指正 .

02 一些减少if-else的编码方式

2.1 方式一 : 提前return

首先展示一段代码示例 :

if (condition1) {
    if (condition2) {
        return getSomething();
    } else {
        return 0;
    }
} else {
    return 0;
}

修改后的代码如下 :

//这里最好对这个flag所判断的逻辑补充注释进行描述
boolean flag = !condition1 || (condition1 && !condition2)
if(flag) {
	return 0;
}

if (condition1 && condition2) {
    return getSomething();
} 

如果存在已知在某些条件下 , 需要返回固定值的逻辑 , 可以将这部分逻辑抽取为一个独立的 if-else block , 并置于其他if-else block的前面 , 当符合该特定条件时 , 直接提前 return 固定值 . 这种方式最直接的效果就是降低if-else的嵌套数量 .

 

2.2 方式二 : 使用三目运算符

先上例子 , 这里以一个业务场景为例 :

查询某条评论的图片URL列表 (如果有 , 评论的图片url列表以JSON数组字符串格式保存在评论表中)

修改前的代码如下 :

Comment comment = getById(commentId);

if (Objects.isNull(comment)) {
    throw new RuntimeException("评论不存在或已被删除");
}

String imgListStr = comment.getImgList();
if(StringUtils.isEmpty(imgListStr)) {
    return null;
}

return JSON.parseArray(imgListStr, String.class);

修改后 :

Comment comment = getById(commentId);

if (Objects.isNull(comment)) {
    throw new RuntimeException("评论不存在或已被删除");
}

String imgListStr = comment.getImgList();
return StringUtils.isEmpty(imgListStr)) ?
    null : JSON.parseArray(imgListStr, String.class);

 

2.3 方式三 : 使用Assert断言

在编写业务代码过程中 , 如果需要对某些特定的条件进行判断 , 且当条件不满足时需要抛出异常 . 对于这种场景 , 除了使用上面三目运算符的示例当中的if方式 , 还可以通过使用Spring Framework 给我们提供的 Assert 工具类进行 .

其中常用的API 有 :

  • isTrue(boolean expression , String message) :expressio == false时 , 会抛出异常 , 异常的message则为第二个入参 ;
  • void notNull(@Nullable Object object, String message) : 同上 , 当object == null 时 , 会抛出异常;
  • void notEmpty(@Nullable Collection<?> collection, String message) : 同上 , 当集合对象为null或者集合元素为空时 , 会抛出异常 .
  • .....

还有其他较多方法 , 可以直接看源码的解析 , 当然实际上isTrue() 已经够用了 , 如果需要更加的语义化 , 可以使用对应的API .

修改前代码 :

if (Objects.isNull(comment)) {
    throw new RuntimeException("评论不存在或已被删除");
}

修改后代码 :

 Assert.isTrue(Objects.nonNull(comment),"评论不存在或已被删除");
 Assert.notNull(comment,"评论不存在或已被删除");

目前Assert工具方法只能抛出单一一种异常 java.lang.IllegalArgumentException , 如果需要自定义所抛出的异常 , 则该方式不适用 .

 

2.4 方式四 : 使用Optional

Optional是 java8 的新特性 , 相当于一个对象的容器 , 主要用于对象的null值校验 , 以及在进行校验后可链式地进行后续操作 , 如 : 抛出异常、null替换 等 .

其中我个人比较常用的几个方法为 :

  • static <T> Optional<T> ofNullable(T value) : 使用Optional 将对象进行包裹 ;
  • T orElse(T other) : Optional中的对象为null时 ,返回入参的对象 .
  • T orElseGet(Supplier<? extends T> other) : Optional中的对象为null时 , 返回Supplier 提供的值 ;
  • T orElseThrow(Supplier<? extends X> exceptionSupplier) : Optional中的对象为null时 , 抛出supplier提供的自定义异常

代码示例 :

Message message1 = Optional.ofNullable(getById(messageId))
    .orElseThrow(() -> new RuntimeException("消息不存在!"));

Message message2 = Optional.ofNullable(getById(messageId))
    .orElse(new Message());

Message message3 = Optional.ofNullable(getById(messageId))
    .orElseGet(Message::new);

由于我日常需要的进行空值判断的比较多的场景是从数据库查询数据完毕时 , 需要查询结果进行空值判断 . 由于我所在的公司使用的持久层框架是mybatis , 不像Spirng Boot 2.x 默认版本的JPA 那样DAO层方法支持返回值为Optional , 所以这里如果需要使用Optional , 只能手动去使用上面列举的第一个方法对查询结果进行包装 .

当然 , IDEA其实已经给我们提供了该包装方式的热键了 , 如下图所示 :

03 结语

个人的一些减少if-else 编码习惯分享就这里了 , 这几种方式里面 , 我个人觉得效果最明显的还是第一种 提前return , 很多时候 , 提前return 也可以很好降低一段代码的复杂度 .

当然如果必须要使用大量的if-else 去控制逻辑时 , 在每个condition 标明一下注释还是一个挺不错的习惯 .


 

### Java 实现钉钉单聊机器人 要通过 Java 开发钉钉单聊机器人,首先需要了解钉钉的 API 文档并按照其规定完成配置和编码工作。以下是关于如何实现的具体方法: #### 1. 创建应用并获取 Access Token 在开发之前,需先登录钉钉开发者后台创建企业内部应用,并获得 AppKey 和 AppSecret。这些参数用于生成 Access Token,这是调用钉钉 OpenAPI 的必要凭证。 Access Token 可以通过 HTTP 请求的方式动态获取: ```java import java.io.BufferedReader; import java.io.InputStreamReader; import java.net.HttpURLConnection; import java.net.URL; public class AccessTokenUtil { public static String getAccessToken(String appKey, String appSecret) throws Exception { URL url = new URL("https://oapi.dingtalk.com/gettoken?appkey=" + appKey + "&appsecret=" + appSecret); HttpURLConnection conn = (HttpURLConnection) url.openConnection(); BufferedReader reader = new BufferedReader(new InputStreamReader(conn.getInputStream())); StringBuilder response = new StringBuilder(); String line; while ((line = reader.readLine()) != null) { response.append(line); } reader.close(); return response.toString(); // JSON 格式的响应数据包含 access_token 字段 } } ``` 此代码片段展示了如何通过指定 `AppKey` 和 `AppSecret` 获取 Access Token[^3]。 #### 2. 调用单聊消息接口 钉钉提供了专门的消息发送接口来支持单聊功能。可以通过以下方式构建请求体并向服务器发起 POST 请求: ```java import com.alibaba.fastjson.JSONObject; public class SingleChatMessageSender { private static final String SEND_URL = "https://oapi.dingtalk.com/chat/send"; public static void sendSingleChatMessage(String accessToken, String chatId, String content) throws Exception { JSONObject body = new JSONObject(); body.put("chatid", chatId); JSONObject msg = new JSONObject(); msg.put("msgtype", "text"); JSONObject textContent = new JSONObject(); textContent.put("content", content); msg.put("text", textContent); body.put("msg", msg); URL url = new URL(SEND_URL + "?access_token=" + accessToken); HttpURLConnection conn = (HttpURLConnection) url.openConnection(); conn.setRequestMethod("POST"); conn.setDoOutput(true); conn.getOutputStream().write(body.toJSONString().getBytes()); int statusCode = conn.getResponseCode(); if (statusCode == 200) { System.out.println("消息已成功发送!"); } else { throw new RuntimeException("失败:" + statusCode); } } } ``` 上述代码实现了向特定聊天 ID 发送文本消息的功能[^4]。 #### 3. 处理安全校验(可选) 如果启用了加签机制,则每次请求都需要附加签名字段作为额外的安全措施。签名计算逻辑如下所示: ```java import javax.crypto.Mac; import javax.crypto.spec.SecretKeySpec; import java.nio.charset.StandardCharsets; import java.util.Base64; public class SignUtils { public static String generateSign(long timestamp, String secret) throws Exception { String stringToSign = timestamp + "\n" + secret; Mac mac = Mac.getInstance("HmacSHA256"); mac.init(new SecretKeySpec(secret.getBytes(StandardCharsets.UTF_8), "HmacSHA256")); byte[] signData = mac.doFinal(stringToSign.getBytes(StandardCharsets.UTF_8)); return Base64.getEncoder().encodeToString(signData); } } ``` 该工具类可用于生成符合要求的时间戳与密钥组合而成的有效签名字符串[^1]。 --- ### 注意事项 - **权限管理**:确保所使用的账号具有足够的操作权限。 - **频率限制**:注意不同类型的 API 存在各自的速率控制策略,请合理安排调用间隔以免触发限流保护。 - **错误排查**:当遇到异常情况时,仔细阅读返回结果中的 error_code 和 errormsg 提示信息以便快速定位问题所在。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值