Android中的NullPointerException

前言

您在编写Android程序时,有没有为经常遇到NullPointerException(空指针异常,简称NPE)而头疼,代码调试过程中,应用运行,跑着跑着就崩溃了,看下调用栈,绝大多数都是因为空指针引起的,即使应用上线后,你收到自己应用异常崩溃数据,也会发现绝大多数都是NPE。此时您可能会责怪自己在写代码时不细心,没有在可能产生异常的地方判断一下。但我想说的是,这并不能怪您,这是Java语言的设计问题,连Java之父(James Gosling)都承认这是他的一项巨大失误。所以接下来我们就探讨下为什么NPE会经常出现,为什么它一直没有解决,我们在编程时应该注意什么。

为什么产生NPE

null是什么,就像每种原始类型都有默认值一样,如int是0,boolean是false。null是java中任何引用类型的默认值。它和public、static、final一样是一个关键字,表示缺失的东西。Java中的引用,有点像是C语言中的指针,因为它也是指向了一个对象,所以Java一旦看到为null的引用被使用,就会报一个运行时错误--NullPointerException。

为什么未解决

Java的引用不完全像C语言中指针的概念,C语言指针可以指向任何内存块,甚至可以越界访问内存,而Java中的引用,你只能指向特定对象,当然也不能在它的范围之外访问,实现此是需要以少量内存开销和运行时的下标检查为代价的,但由此换来的安全和效率的提高是很值得的。Java中的引用存在于栈区,所分配的对象在堆区,Java的内存回收机制会根据算法清理堆区无用的对象,比如这个对象没有对应的引用指向它,它就可能会被回收。如果一个对象被回收后,您再使用这个对象,就会产生NPE。比如Android中很臭名昭著的Activity被回收,而在Fragment中获得上下文就会出现NPE的问题。

我们要怎么做

我们可以在可能会产生NPE的地方做下判断,比如上文提到的Activity被回收问题,我们可以在fragment中要使用上下文的地方加一个isAdded判断一下Activity是否存在再来使用。当然您也可以期待Android可以尽快支持Java 8,因为Java 8中引入了一个Optional机制可以用来解决NPE。详情可以查看:《Tired of Null Pointer Exceptions? Consider Using Java SE 8's Optional
!》。但不幸的是,根据上次IO大会时Android开发团队的回答,Android要支持Java 8,近期是无法实现的了(最新消息,Android N的preview版本发布了,并且这一版本中支持了Java 8)。当然您也可以尝试使用kotlin语言,kotlin语言可以有效的防止NPE问题。这里有关于kotlin介绍的文章还不错:《Kotlin, the Swift of Android》

最后推荐下:《Java编程思想》Bruce Eckel著,大师的作品每一句话都耐人寻味。

你可能感兴趣的:(Android中的NullPointerException)