为什么阿里巴巴说要对switch括号内的String变量进行判空?

引言

空指针NullPointerException可以说是开发中最经常碰到的异常了。良好的代码规范可以规避掉一些这样的异常。在《阿里巴巴规范手册》中有这样一条规范:

我们可以从这句话得出两个前置条件:

  1. switch括号内的变量类型为String。
  2. 此变量为外部参数时。

变量类型为String

你知道这行代码输出什么吗?

java 复制代码
public class SwitchString {

    public static void main(String[] args) {
        method (null);
    }

    public static void method(String param) {
        switch (param) {
            // 肯定不是进入这里
            case "sth":
                System.out.println("it's sth");
                break;
            // 也不是进入这里
            case "null":
                System.out.println("it's null");
                break;
            // 也不是进入这里
            default:
                System.out.println("default");
        }
    }
}

如果我们不看上面的注释,你会认为程序会打印出哪句话呢?我们点击运行,报了NullPointerException空指针异常~

txt 复制代码
Exception in thread "main" java.lang.NullPointerException
	at SwitchString.method(SwitchString.java:8)
	at SwitchString.main(SwitchString.java:4)

很奇怪,为啥会空指针呢? 你可能会说第8行param为空,所以空指针了。虽然这样理解也没问题。

我们要知道,NPE异常是代码中使用了这个对象的某些方法或者属性,而这个对象为空所以就会抛出异常。例如以下代码

java 复制代码
public class TestNPE {
    private int id;

    public static void main(String[] args) {
        TestNPE testNPE = null;
        System.out.println(testNPE);
        System.out.println(testNPE.id);
    }
}

我们点击运行可以得知。虽然testNPE这个对象为null,但是我们对其打印是没有问题的,但如果涉及到了对这个对象的某些方法或者属性的调用就会报错。

txt 复制代码
null
Exception in thread "main" java.lang.NullPointerException
	at TestNPE.main(TestNPE.java:8)

而我们上面那个例子,它是在哪里使用了这个对象呢?

其实swtich是Java中的一个语法糖。(更多详情可以看以前的文章juejin.cn/post/729516...

JDK7之前switch本身只支持基本类型,JDK7开始支持String。

Java中的switch自身原本就支持基本类型。比如intchar等。对于int类型,直接进行数值的比较。对于char类型则是比较其ascii码。所以,对于编译器来说,switch中其实只能使用整型,任何类型的比较都要转换成整型。比如byteshortchar(ackii码是整型)以及int

我们将上面的代码进行反编译可以得到

java 复制代码
public class SwitchString {

   public static void main(String[] args) {
      method((String)null);
   }

   public static void method(String param) {
      byte var2 = -1;
      switch(param.hashCode()) {
      case 114215:
         if(param.equals("sth")) {
            var2 = 0;
         }
         break;
      case 3392903:
         if(param.equals("null")) {
            var2 = 1;
         }
      }

      switch(var2) {
      case 0:
         System.out.println("it's sth");
         break;
      case 1:
         System.out.println("it's null");
         break;
      default:
         System.out.println("default");
      }

   }
}

字符串的switch是通过equals()hashCode()方法来实现的。 由于调用了这两个方法,而param为null,所以出现了空指针异常。

外部参数

那么为啥还要是外部参数呢?其实很简单,我们把代码修改一下:

java 复制代码
public class SwitchString {

    public static void main(String[] args) {
        switch (null) {
            // 肯定不是进入这里
            case "sth":
                System.out.println("it's sth");
                break;
            // 也不是进入这里
            case "null":
                System.out.println("it's null");
                break;
            // 也不是进入这里
            default:
                System.out.println("default");
        }
    }
}

这样的代码是会报错提示的且编译是无法通过。~~

总结

我们对switch这种情况进行了分析,得出了两个条件:

  1. switch括号内的变量类型为String。
  2. 此变量为外部参数时。

条件1是String类型的变量参数对象的判断是通过equals()hashCode()方法来实现的。所以变量参数为null,调用这两个方法就会出现NPE。

条件2则是编译器会帮我们检查并且这是不符合语法的不能进行编译。

相关推荐
m0_5719575841 分钟前
Java | Leetcode Java题解之第543题二叉树的直径
java·leetcode·题解
魔道不误砍柴功3 小时前
Java 中如何巧妙应用 Function 让方法复用性更强
java·开发语言·python
NiNg_1_2343 小时前
SpringBoot整合SpringSecurity实现密码加密解密、登录认证退出功能
java·spring boot·后端
闲晨3 小时前
C++ 继承:代码传承的魔法棒,开启奇幻编程之旅
java·c语言·开发语言·c++·经验分享
Chrikk4 小时前
Go-性能调优实战案例
开发语言·后端·golang
幼儿园老大*4 小时前
Go的环境搭建以及GoLand安装教程
开发语言·经验分享·后端·golang·go
canyuemanyue4 小时前
go语言连续监控事件并回调处理
开发语言·后端·golang
杜杜的man4 小时前
【go从零单排】go语言中的指针
开发语言·后端·golang
测开小菜鸟4 小时前
使用python向钉钉群聊发送消息
java·python·钉钉
P.H. Infinity5 小时前
【RabbitMQ】04-发送者可靠性
java·rabbitmq·java-rabbitmq