Java中通过反射实现代理Proxy代码实例
1.实现一个简易的代理类
java实现代理可以通过java.lang.reflect.Proxy接口结合java.lang.reflect.InvocationHandler来实现
import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; public class ProxyHandler implements InvocationHandler { private UserService object; public ProxyHandler() { this.object = new UserServiceImpl(); } public UserService newProxyInstance(){ return (UserService) Proxy.newProxyInstance(this.object.getClass().getClassLoader(), this.object.getClass().getInterfaces(), this); } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("预处理"); Object object = method.invoke(this.object, args); if(object instanceof Person) { ((Person) object).setName("姓名:" + ((Person) object).getName()); } System.out.println("后处理"); return object; } }
通过如下方法调用生成代理的对象
public class Test { public static void main(String[] args) { ProxyHandler proxyHandler = new ProxyHandler(); //生成代理类初始化被代理类 UserService userService = proxyHandler.newProxyInstance(); //生成代理实例 System.out.println(userService.getUserInfo(1).getName()); //调用被代理方法 } }
此时代理类内自定义的invoke方法将会在被代理方法执行前后执行自定义的操作
控制台输出结果
预处理
后处理
姓名:libiyi
2.Proxy.newProxyInstance()解析
如下是源码
public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) throws IllegalArgumentException { Objects.requireNonNull(h); final Class<?>[] intfs = interfaces.clone(); final SecurityManager sm = System.getSecurityManager(); if (sm != null) { checkProxyAccess(Reflection.getCallerClass(), loader, intfs); } /* * 从缓存中获取到代理类 */ Class<?> cl = getProxyClass0(loader, intfs); try { if (sm != null) { checkNewProxyPermission(Reflection.getCallerClass(), cl); } final Constructor<?> cons = cl.getConstructor(constructorParams);//获取到代理类的构造器 final InvocationHandler ih = h; if (!Modifier.isPublic(cl.getModifiers())) { AccessController.doPrivileged(new PrivilegedAction<Void>() { public Void run() { cons.setAccessible(true); return null; } }); } return cons.newInstance(new Object[]{h});//生成代理类实例 } catch (IllegalAccessException|InstantiationException e) { throw new InternalError(e.toString(), e); } catch (InvocationTargetException e) { Throwable t = e.getCause(); if (t instanceof RuntimeException) { throw (RuntimeException) t; } else { throw new InternalError(t.toString(), t); } } catch (NoSuchMethodException e) { throw new InternalError(e.toString(), e); } }
从如上代码可见,首先需要获取代理类,然后再获取到代理类构造器,最后使用构造器生成代理类实例。
关键的代理类获取在getProxyClass0方法、和getConstructor中
private static Class<?> getProxyClass0(ClassLoader loader, Class<?>... interfaces) { if (interfaces.length > 65535) { throw new IllegalArgumentException("interface limit exceeded"); } // If the proxy class defined by the given loader implementing // the given interfaces exists, this will simply return the cached copy; // otherwise, it will create the proxy class via the ProxyClassFactory return proxyClassCache.get(loader, interfaces); } private static final WeakCache<ClassLoader, Class<?>[], Class<?>> proxyClassCache = new WeakCache<>(new KeyFactory(), new ProxyClassFactory());
在getProxyClass0方法中,使用了关键的WeakCache类,其中的结构是ConcurrentMap<Object, ConcurrentMap<Object, Supplier<V>>>。 最外层map的key是通过classLoader得出的CacheKey类型,value是一个map,其中key为由classLoader和interfaces[]共同得出的一个Key,value是缓存的代理类。
到此这篇关于Java中通过反射实现代理Proxy代码实例的文章就介绍到这了,更多相关Java反射实现代理内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
相关文章
J2EE Servlet上传文件到服务器并相应显示功能的实现代码
这篇文章主要介绍了J2EE Servlet上传文件到服务器,并相应显示,在文中上传方式使用的是post不能使用get,具体实例代码大家参考下本文2018-07-07SpringBoot mybatis-plus使用json字段实战指南
在现代应用开发中经常会使用JSON格式存储和传输数据,为了便捷地处理数据库中的JSON字段,MyBatis-Plus提供了强大的JSON处理器,这篇文章主要给大家介绍了关于SpringBoot mybatis-plus使用json字段的相关资料,需要的朋友可以参考下2024-01-01
最新评论