读者被问题代码折磨,鸭哥劝 NullPointerException 耗子尾汁!!!

简介: 说起来,鸭哥也算是身经百战的码农了,代码习惯和风格都是不错的。没想到,今天大意了没有闪,NullPointerException 就找上门来了。

image.png

看完问题鸭哥想说,NullPointerException 确实不讲武德,小伙子大意了没有闪,鸭哥来帮你解决~

题目如下:

//不讲武德的代码
public class Printer {
    private String name;
    public void setName(String name) {
        this.name = name;
    }
    public void print() {
        printString(name);
    }
    private void printString(String s) {
        System.out.println(s + " (" + s.length() + ")");
    }
    public static void main(String[] args) {
        Printer printer = new Printer();
        printer.print();
    }
}

运行结果:

Exception in thread "main" java.lang.NullPointerException
    at Printer.printString(Printer.java:13)
    at Printer.print(Printer.java:9)
    at Printer.main(Printer.java:19)


根据报错信息(stacktrace),可以顺藤摸瓜,找出问题所在:

  1. 程序的编译没有问题,编译后运行,线程"main"抛出了 NullPointerException 空指针异常;
  2. 继续往下看,新建的对象 printer 调用 printString 方法时,第 13 行代码抛出了异常,出现问题的代码是  s.length() ;
  3. 执果索因,printString 方法中,s 的值从何而来呢?是在 print() 方法中通过 调用printString(name) 传入的。检查后可以发现,问题在于 this.name 为 null;
  4. 问题找到了,由于新建对象时没有初始化,调用了 null 的实例方法,导致抛出了异常。


正确的代码如下:


public class Printer {
    private final String name;
    public Printer(String name) {
        this.name = Objects.requireNonNull(name);
    }
    public void print() {
        printString(name);
    }
    private void printString(String s) {
        System.out.println(s + " (" + s.length() + ")");
    }
    public static void main(String[] args) {
        Printer printer = new Printer("123");
        printer.print();
    }
}


故事到这里似乎就结束了,但俗话说吃一堑长一智,鸭哥还是要继续扒一下 NullPointerException 。


在Java中,当我们用如下方式声明一个基本数据类型的变量时。


int x;
x = 1;


第一行,声明了 int 型变量 x,且 Java 会将 x 的值初始化为 0;


第二行,将 10 赋值给 x,10这个数值会被写入变量 x 指向的内存中。但当我们用类似的方式声明一个引用数据类型的对象时,Java执行的操作并不相同


Integer num;
num = new Integer(1);


第一行,声明了 num 变量,但因为 Integer 属于引用类型,此时 num 并不包含数值,而是一个指针,并且指针的值为 null, 即空指针,不指向任何地址;


第二行,先是创建了一个 Integer 对象,然后将 num 变量指向了该对象。


当我们声明了一个引用类型的变量,却没有创建指向该变量的对象,此时访问这个变量的内容或调用它的实例方法,就会抛出 NullPointerException (NPE)  异常,因为我们要访问的内容实际上并不存在。


常见的处理方式是,先判断参数是否为空,非空时再进行接下来的处理,模式如下:


public void doSomething(someObject obj) {
    if(obj == null) {
       // Do something
    } else {
       // Do something else
    }
}


最后,鸭哥想说,应对 NullPointerException (NPE)  的最有效方法就是:


所有不是我们自己创建的对象,一律检查该对象是否为空


如果是调用者给某方法传入了无效参数,最好将抛出的异常返回给调用者。忽略无效参数且不做任何处理是最不可取的,因为这样做隐藏了问题,后续出错时再排查会相当困难。


关于 NullPointerException,你有哪些注意事项想要分享给大家呢?


相关文章
|
1月前
|
算法 程序员
10年老程序员告诉大家什么时候该写什么样的代码
10年老程序员告诉大家什么时候该写什么样的代码
23 0
|
8月前
|
人工智能 程序员 vr&ar
竟有程序员40了还在撸代码?
竟有程序员40了还在撸代码?
103 2
|
9月前
|
设计模式 缓存 算法
花了30天才肝出来,史上最全面Java设计模式总结,看完再也不会忘
Design Patterns: Elements of Reusable Object-Oriented Software(以下简称《设计模式》),一书由Erich Gamma、Richard Helm、Ralph Johnson和John Vlissides合著(Addison-Wesley,1995)。这四位作者常被称为“四人组(Gang of Four)”,而这本书也就被称为“四人组(或 GoF)”书。他们首次给我们总结出一套软件开发可以反复使用的经验,帮助我们提高代码的可重用性、系统的可维护性等,解决软件开发中的复杂问题。
123 0
|
消息中间件 存储 JavaScript
如何写出一手让同事膜拜的漂亮代码?
如何写出一手让同事膜拜的漂亮代码?
|
前端开发 Java Spring
求求你不要写满屏的 try...catch 了,这才是优雅的处理方式,真香...
求求你不要写满屏的 try...catch 了,这才是优雅的处理方式,真香...
238 0
求求你不要写满屏的 try...catch 了,这才是优雅的处理方式,真香...
|
Java API 容器
遇到空指针(npe)切莫慌 ✨ jdk 8中有妙招 ✨ 每日积累
遇到空指针(npe)切莫慌 ✨ jdk 8中有妙招 ✨ 每日积累
遇到空指针(npe)切莫慌 ✨ jdk 8中有妙招 ✨ 每日积累
|
安全 Java 编译器
学妹不懂Java泛型,非让我写一篇给她看看(有图为证)
笔者有个学妹就遇到了相同的境遇,学弟被泛型搞得头晕目眩,搞不懂泛型是个啥玩意。天天用的泛型也不知道啥玩意(她可能都不知道她有没有用泛型)。立图为证!当然,笔者深度还欠缺,如果错误还请指正!
117 0
学妹不懂Java泛型,非让我写一篇给她看看(有图为证)
|
Python
又烧脑又炫技还没什么用,在代码里面打印自身
又烧脑又炫技还没什么用,在代码里面打印自身
174 0
又烧脑又炫技还没什么用,在代码里面打印自身
|
安全 Java
老爷子这代码,看跪了! (下)
老爷子这代码,看跪了! (下)
108 0
老爷子这代码,看跪了! (下)
|
Java
老爷子这代码,看跪了! (上)
老爷子这代码,看跪了! (上)
130 0
老爷子这代码,看跪了! (上)

相关实验场景

更多