aop动态代理底层实现模拟
UserDAO.java:目标方法
1 2 3 4 5 6 7 8 9 10 | public class UserDAO implements IDAO{ @Override public void save() { // TODO Auto-generated method stub } @Override public void select () { // TODO Auto-generated method stub } } |
Proxy.java:代理类
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 | import java.io.File; import java.io.FileInputStream; import java.io.FileWriter; import java.lang.reflect.Method; import javax.tools.JavaCompiler; import javax.tools.JavaCompiler.CompilationTask; import javax.tools.StandardJavaFileManager; import javax.tools.ToolProvider; public class Proxy { public static void main(String[] args) throws Exception { Object proxy=newProxyInstance(); System. out .println(proxy.toString()); } //传进来方法信息(来自接口,也可以来之目标类) //类加载器 //传进来一个invocationHandler的实现类 public static Object newProxyInstance() throws Exception { //模拟代理对象的创建; // 得到方法信息 Class classInfo = IDAO. class ; Method[] methods = classInfo.getMethods(); for (Method method : methods) { System. out .println(method.getName()); } // 生成Proxy1.java String method1Name = methods[0].getName(); String method2Name = methods[1].getName(); //代理类 save(){调用接口InvocationHandler.invoke()} String classContent = "public class Proxy1" + "{public void " +method1Name+ "() {} " + "public void " +method2Name+ "(){}" + "}" ; String fileString = "d:\\Proxy1.java" ; FileWriter fileWriter = new FileWriter(fileString); fileWriter.write(classContent); fileWriter.flush(); // 编译出Proxy1.class JavaCompiler compiler = ToolProvider.getSystemJavaCompiler(); StandardJavaFileManager fileManager = compiler.getStandardFileManager( null , null , null ); File file = new File(fileString); Iterable units = fileManager.getJavaFileObjects(file); CompilationTask task = compiler.getTask( null , fileManager, null , null , null , units); task.call(); // 加载Proxy1.class ProxyClasLoader clasLoader= new ProxyClasLoader(); Class proxyClassInfo=clasLoader.findClass( "d:\\Proxy1.class" ); // 创建出代理对象,返回 Object proxy=proxyClassInfo.newInstance(); return proxy; } static class ProxyClasLoader extends ClassLoader{ @Override protected Class<?> findClass(String name) throws ClassNotFoundException { try { FileInputStream inputStream= new FileInputStream(name); int length=inputStream.available(); byte [] classContent= new byte [length]; inputStream.read(classContent); Class classInfo=defineClass(classContent, 0, classContent.length); return classInfo; } catch (Exception e) { // TODO: handle exception } return null ; } } } |
其中newProxyInstance方法在实际中由java.lang.reflect.Proxy类中实现创建;
newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)
//类加载器,目标类的方法信息;InvocationHandler用来代理方法的调用;//整个过程用来创建代理对象
IDAO.java:目标方法实现的接口
1 2 3 4 5 | public interface IDAO { public void save(); public void select (); } |
图解思路:
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】博客园社区专享云产品让利特惠,阿里云新客6.5折上折
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 用 C# 插值字符串处理器写一个 sscanf
· Java 中堆内存和栈内存上的数据分布和特点
· 开发中对象命名的一点思考
· .NET Core内存结构体系(Windows环境)底层原理浅谈
· C# 深度学习:对抗生成网络(GAN)训练头像生成模型
· 为什么说在企业级应用开发中,后端往往是效率杀手?
· 本地部署DeepSeek后,没有好看的交互界面怎么行!
· 趁着过年的时候手搓了一个低代码框架
· 用 C# 插值字符串处理器写一个 sscanf
· 推荐一个DeepSeek 大模型的免费 API 项目!兼容OpenAI接口!