在做简单的MVC项目的时候。在dao里莫名其妙的遇到NullPointerException。赶时间的朋友直接看 摘要 或 正文的倒数第二句话~
背景知识:
1、Exception:是程序员在编码过程中,如果出现某种情况(比如文件不存在FileNotFoundException,类型转换错误ClassCastException 等),就可以通过throw new FileNotFoundException();来说明,此时当前调用的异常信息(哪个类什么方法第几行)会记录到异常栈中。也就是说,异常是编程人员手动抛出的,并且异常栈中会记录异常信息。因此遇到异常,一般通过异常栈定位发生异常的位置,点进方法找throw new xxxException();,接下来就不用我说了。
2、NullPointerException:是在调用某个类的方法时,发现这个类是null则抛出NullPointerException,如下
43 Book book = new Book();
44 String str = book.getPublisher();
45 System.out.println(str);
46 System.out.println(str.substring(0));
执行第二行代码后,str=null;,因此,在第四行代码执行时,抛出NullPointerException,抛出的异常信息如下。
Exception in thread "main" java.lang.NullPointerException
at com.jansing.first.main(first.java:46)
然而,有一种NullPointerException不遵守上面的规则。。。
本次遇到的问题:
代码如下,在dao层调用session.createSQLQuery().setLong().executeUpdate();抛出空指针异常,确定是role.getId()返回null导致。
65 public void addPermission(Role role, Long[] pids) throws Exception{
66 Session session = getSession();
67 //在setLong("rid", role.getId())这里,role.getId()=null
68 session.createSQLQuery("delete from role_permission where rid=:rid")
69 .<span style="color:#ff0000;">setLong("rid", role.getId())</span>.executeUpdate();
70 }
java.lang.NullPointerException
at com.etop.info.dao.RoleDao.addPermission(RoleDao.java:69)
at com.etop.info.service.RoleService.saveOrUpdate(RoleService.java:28)
看到问题了没有?在异常栈的最顶层,他竟然说我69行代码空指针。还没看懂的话,对比背景知识2再想1s,说不定懂了呢。
没错,异常栈中没有真正的异常信息,即异常在哪个类什么方法第几行发生,而最顶层的RoleDao.addPermission(RoleDao.java:69)并没有使用到role.getId()得来的null对象的任何属性或方法,所以异常不应该从这里抛出。
这让你想到了什么?反正我想到了能美征二。。。orz
明明存在,对战列表却木有。。。T^T
为表达对此的愤慨,作此文。以下为解决过程。
解决过程:
因为role.getId()返回null,初步确定是setLong()或executeUpdate()中使用了这个null值的属性/方法。点进这两个方法,给方法内第一行代码加断点,准备查看哪行出异常。然而运行项目后,发现,断点没起作用,就已经抛异常了。也就是说,在调用这两个方法前,就抛了异常。最后瞄了眼setLong的方法签名,整个人都不好了。
public Query setLong(String name, long val);
第二个参数为long型,而role对象的属性id是Long型。所以这次问题实际上是调用方法时包装类的自动拆箱导致的异常(留坑)。
至此,能美讨伐战惨胜。。。