-
为什么我们应该避免NullPointerException
在Android开发中,最常见的就是NullPointerException,然而这个异常最致命便是导致了APP闪退,如果你在开发过程中没有过多的关注对象是否为null而轻易的调用其中的方法,那APP的闪退则很大程度是因为你没有检查null情况。然而在繁忙的开发过程中,导致我们每次使用引动都必须测试是否为Null,在代码上枯燥且冗余。问题在于null 除了在你试图用它执行任何操作来产生NullPointerException之外,它自己没有任何行为,这让我们很沮丧。
-
何为空对象概念
空对象最有用之处在于它更靠近数据,因为对象表示的是问题空间的实体。在接下来的例子中,都有一个Student类,而在代码里面,有很多情况目前你都没有一个实际的Student,通常你会使用null引用并且测试它。与此不同的是,我们仍然使用空对象。但是这个空对象可以响应“实际”对象响应的所有消息,但是你仍然需要以某种方式测试它是否为空。最简单的方式是创建一个标记借口:
public interface Null()
首先我们先来一个最常见的学生类
public class Student implements Null{ private String name; private String age; private String major; public Student(String name, String age, String major) { this.name = name; this.age = age; this.major = major; } @Override public String toString() { return "Student{" + "name='" + name + '\'' +", age='" + age + '\'' +", major='" + major + '\'' +'}'; } public static class NULLStudent extends Student implements Null{ private NULLStudent(){ super("","","");} @Override public String toString() { return "Nullperson"; } } public static final Student NULL = new NULLStudent(); }
-
我们如何创建空对象类以及如何处理空对象
通常,空对象都是单例,因此这里作为静态final实例创建。这可以正常工作,如果你想要修改一个NullStudent,那只能用一个新的对象来替换它。此处你可以用instanceof来探测是Null 还是更具体的NullStudent,但是此处我们使用了单例方式,所有你甚至可以用equals和==来和Student.Null来比较。
接下来我们来编写一个Position类,其中涉及到当我们没有Student对象时如何来处理。
public class Position { private String titile; private Student student; public Position(String titile, Student student) { this.titile = titile; this.student = student; if(this.student == null){ this.student = Student.NULL; } } public Position(String titile) { this.titile = titile; this.student = Student.NULL; } public String getTitile() { return titile; } public void setTitile(String titile) { this.titile = titile; } public Student getStudent() { return student; } public void setStudent(Student student) { this.student = student; if(this.student == null){ this.student = Student.NULL; } } @Override public String toString() { return "Position{" +"titile='" + titile + '\'' +", student=" + student +'}'; } }
有了Positon类,此时你就不需要创建空对象了,因为Student.Null的存在就表示一个空的Position。
现在我们编写一个Staff来实现填充职位时查询空对象。
public class Staff extends ArrayList<Position>{ public void add(String titile,Student student) { add(titile,student); } public void add(String... titles) { for (String title: titles) { add(new Position(title)); } } public Staff(String... titles) { add(titles); } public boolean positionAvailable(String title) { for (Position position : this) { if(position.getTitile().equals(title) && position.getStudent() == Student.NULL) { return true; } } return false; } public void fillPosition(String title,Student hire) { for (Position position : this) { if(position.getTitile() == title && position.getStudent() == Student.NULL) { position.setStudent(hire); return; } } throw new RuntimeException("Position" + title + "not available"); } public static void main(String[] agrs){ Staff staff = new Staff("CTO","Marketing Manager","SoftWare Engineer","Test Engineer","Technical writer"); staff.fillPosition("CTO",new Student("ME","24","CS")); staff.fillPosition("SoftWare Engineer",new Student("Another","21","unKonw")); if(staff.positionAvailable("Technical writer")) { staff.fillPosition("Technical writer",new Student("Finch","18","City")); } System.out.println(staff); } }
-
最后关于空对象我想说的
注意 你在某些地方仍然必须测试他是否为空对象,但你永远不会引用到null,取而代之的则是我们编写的NULL对象(空的数据实体)。在某种程度上我们能确保你不会因为无意间引用到null而导致NullPointerException。进而导致令人头大的APP 闪退的问题。
最后我想说的是, 关于空和无的区别。 空在某种程度上是占据空间的实体,然而无则不是。对应这篇文章想说的则是,如果在之前我们常用的null 则可以对应无。然而我们编写的NULL类则可以对应空,它可以作为一个数据实体存在,只是它作为一个数据实体暂时里面数据为空而已。如果可以我们应该尽量避免只是使用null,这是C语言留下的习惯,但是我们在JAVA完全可以避免这些。
-
PS 关于Collection里面的空对象
在我们编写我们对象实体数据类的时候,我们可以采取这种方式来规避Null,其实在容器类中,Collections工具类以及提供静态方法返回相关的空容器返回,所以下次从网络获取数据为空的时候完全可以返回一个不可以改变的空容器。用法如下:
Collections.EMPTY_LIST
空的列表(不可变的)。
Collections.EMPTY_MAP
空的映射(不可变的)。
Collections.EMPTY_SET
空的 set(不可变的)。