一、前言
在学习JDK提供的动态代理时,写了个小demo,但是发现执行有些方法抛出空指针异常,有些方法不会。
二、实现代码
接口:
package test;
/**
* 被代理的接口
* @author Satone
* @date 2019年1月25日上午8:03:53
*/
public interface UserDao {
void findAll();
}
代理类:
package test;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/**
* 代理UserDao接口
* @author Satone
* @date 2019年1月25日上午8:05:22
*/
public class UserProxy {
/**
* 返回特殊代理类的对象,该对象代理接口中的抽象方法,包括Object中默认实现的非final方法
* @param clazz
* @return
*/
public Object getMapper(Class clazz) {
return Proxy.newProxyInstance(clazz.getClassLoader(), new Class[]{clazz}, new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("hello, world");
return null;
}
});
}
public static void main(String[] args) {
UserProxy proxy = new UserProxy();
UserDao mapper = (UserDao)proxy.getMapper(UserDao.class);
// 正常执行,不会空指针
System.out.println(mapper.toString());
// 抛出空指针异常
System.out.println(mapper.hashCode());
mapper.findAll();
// mapper对象为代理类实例,类名class com.sun.proxy.$Proxy0
System.out.println(mapper.getClass());
}
}
三、原因
Proxy.newProxyInstance()方法返回的对象属于一个特殊类,该类不仅代理了接口中的findAll()方法,还代理了Object默认实现的非final方法。当代码执行到:
System.out.println(mapper.hashCode());
hashCode()需要一个int的返回,而在代理时返回了一个值为null的Integer对象,这时进行自动拆箱:
((Integer)null).intValue();
异常就来源于上面这句自动拆箱!而对于toString方法来说,(String)null 是没有任何问题且能执行的,除非显示调用toString()方法:
// 不会空指针,因为可以做到 (String)null
System.out.println((String)null);
// 显示调用null对象的方法,抛出空指针异常
System.out.println(((String)null).toString());