当前位置: 首页 > news >正文

java代理模式

代理模式

为什么要学习代理模式?因为这是SpringAOP的底层! 【SpringAOP和SpingMVC}】

代理模式的分类:

  • 静态代理

  • 动态代理

代理就像这里的中介,帮助你去做向房东租房,你不能直接解出房东,而房东和中介是因为租房而联系在一起,所以租房是接口

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-6pVz2Ts2-1677422310317)(java静态代理.assets/image-20230226102101840.png)]

静态代理

角色分析:

  • 抽象角色(租房):一般使用接口和抽象类来解决
  • 真实角色(房东):被代理的角色
  • 代理角色(中介):代理真实角色,被代理真实角色后,我们一般会做一些附属工作
  • 客户:访问代理对象的人

代码步骤:

  1. 接口

    package com.hong;//租房
    public interface Rent {public void rent();
    }
  2. 真实角色

    package com.hong;public class Client {public static void main(String[] args) {//房东要租房子Host host = new Host();//代理,中介帮房东租房子,但是呢房东会加一些附属操作!
    //        host.rent();Proxy proxy = new Proxy(host);//你不用面对房东,直接找中介即可proxy.rent();}
    }
  3. 代理角色

    package com.hong;public class Proxy implements Rent{private Host host;public Proxy(Host host) {this.host = host;}public Proxy() {}@Overridepublic void rent() {host.rent();seeHost();fare();hetong();}//看房public void seeHost(){System.out.println("中介带你看房");}public void hetong(){System.out.println("签合同");}//收中介费public void fare(){System.out.println("收中介费");}
    }
  4. 客户端访问代理角色

    package com.hong;//房东
    public class Host implements Rent{@Overridepublic void rent() {System.out.println("房东要出租房子");}
    }

代理模式的好处:

  • 可以使真实角色的操作更加纯粹!不用去关注一些公共的业务
  • 公共也就就交给代理角色!实现了业务的分工!
  • 公共业务发生扩展的时候,方便集中管理!

缺点:

  • 一个真实角色就会产生一个代理角色;代码量会翻倍,开发效率会变低

AOP

用代理增加功能,不改动原有的代码,只改动代理类

package com.hong;//真实对象
public class UserServiceImpl implements UserService {@Overridepublic void add() {System.out.println("增加一个用户");}@Overridepublic void delete() {System.out.println("删除一个用户");}@Overridepublic void update() {System.out.println("修改了一个用户");}@Overridepublic void query() {System.out.println("查询了一个用户");}
}
package com.hong;public class UserServiceProxy implements UserService{private UserServiceImpl userservice;public void setUserservice(UserServiceImpl userservice) {this.userservice = userservice;}@Overridepublic void add() {log("add");userservice.add();}@Overridepublic void delete() {log("delete");userservice.delete();}@Overridepublic void update() {}@Overridepublic void query() {}public void log(String msg){System.out.println("调用"+msg+"方法");}
}
package com.hong;public interface UserService {public void add();public void delete();public void update();public void query();
}
package com.hong;public class Client {public static void main(String[] args) {UserServiceImpl userService = new UserServiceImpl();UserServiceProxy proxy = new UserServiceProxy();proxy.setUserservice(userService);proxy.add();}
}

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-FhQMtFQi-1677422310319)(java静态代理.assets/image-20230226202716742.png)]

动态代理

  • 动态代理和静态代理角色一样
  • 动态代理类是动态生成的,不是我们直接写好的
  • 动态代理分为两大类:基于接口的动态代理,基于类的动态代理
    • 基于接口—JDK动态代理
    • 基于类:cglib
    • java字节码实现:javasist

需要了解两个类:proxy:代理,lnvocationHandler:调用处理程序

静态代理的好处:

  • 可以使真实角色的操作更加纯粹!不用去关注一些公共的业务
  • 公共也就就交给代理角色!实现了业务的分工!
  • 公共业务发生扩展的时候,方便集中管理!
  • 一个动态代理类代理的是一个接口,一般就是对应的一类业务
  • 一个动态代理,修改的成本极低

实现代码

动态代理

package com.hong.demo02;import com.hong.xxh.Rent;import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;//自动生成代理
public class ProxyInvocationHandler implements InvocationHandler {//被代理的接口private Object taget;public void setTaget(Object taget) {this.taget = taget;}//生成得到代理对象public Object getProxy() {return Proxy.newProxyInstance(this.getClass().getClassLoader(),taget.getClass().getInterfaces(),this);}//处理代理实例并返回结果public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {Object result = method.invoke(taget, args);return result;}
}

接口

package com.hong.demo02;public interface UserService {public void add();public void delete();public void update();public void query();
}

真实对象

package com.hong.demo02;//真实对象
public class UserServiceImpl implements UserService {@Overridepublic void add() {System.out.println("增加一个用户");}@Overridepublic void delete() {System.out.println("删除一个用户");}@Overridepublic void update() {System.out.println("修改了一个用户");}@Overridepublic void query() {System.out.println("查询了一个用户");}
}

客户端访问

package com.hong.demo02;import com.hong.Proxy;
import com.hong.UserServiceImpl;public class ClenClient {public static void main(String[] args) {//真实角色UserServiceImpl userService = new UserServiceImpl();//代理角色ProxyInvocationHandler pih = new ProxyInvocationHandler();pih.setTaget(userService);  //设置要代理的对象//动态生成代理UserService proxy = (UserService) pih.getProxy();proxy.delete();}
}

反序列化中动态代理的利用

我们将入口类定义为A,我们最理想的情况是 A[O] -> O.f,那么我们将传进去的参数O替换为B即可。但是在实战的情况下这种情况是极少的。

回到实战情况,比如我们的入口类A存在O.abc这个方法,也就是 A[O] -> O.abc;而 O 呢,如果是一个动态代理类,Oinvoke方法里存在.f的方法,便可以漏洞利用了,我们还是展示一下。

A[O] -> O.abc O[O2] invoke -> O2.f // 此时将 B 去替换

O2 最后 ----> O[B] invoke -> B.f // 达到漏洞利用效果

就像readObject在反序列化中自动调用,invoke在动态代理的时候也自动调用,如果我们在invoke插入恶意代码,就可进行攻击例如

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-XAkzDHfq-1677422310320)(java静态代理.assets/image-20230226223422071.png)]
结果
在这里插入图片描述

http://www.lryc.cn/news/21284.html

相关文章:

  • JUC包:CountDownLatch源码+实例讲解
  • Log4j2基本使用
  • A2L在CAN FD总线的使用
  • Android JetPack之启动优化StartUp初始化组件的详解和使用
  • [11]云计算|简答题|案例分析|云交付|云部署|负载均衡器|时间戳
  • C++11/C++14:lambda表达式
  • 算法课堂-分治算法
  • 操作系统权限提升(十六)之绕过UAC提权-CVE-2019-1388 UAC提权
  • 实例9:四足机器人运动学正解平面RR单腿可视化
  • 堆的基本存储
  • 如何获取物体立体信息通过一个相机
  • 【数据挖掘实战】——中医证型的关联规则挖掘(Apriori算法)
  • 一些硬件学习的注意事项与快捷方法
  • 【Tomcat】Tomcat安装及环境配置
  • 负载均衡:LVS 笔记(二)
  • SEO优化:干货技巧分享,包新站1-15天100%收录首页
  • JavaWeb测试题
  • Java EE|TCP/IP协议栈之数据链路层协议详解
  • Lighthouse组合Puppeteer检测页面
  • 【C++】仿函数、lambda表达式、包装器
  • 二叉树(二)
  • 爬虫知识简介
  • 2023年全国最新会计专业技术资格精选真题及答案6
  • 同时学习C++语言和C#语言好吗?
  • Android8,source与lunch流程解析
  • 大数据NiFi(二十):实时同步MySQL数据到Hive
  • mac 如何设置 oh my zsh 终端terminal 和添加主题powerlevel10k
  • 王道《操作系统》学习(一)——计算机系统概述
  • 什么是自适应平台服务?
  • QML Image and Text(图像和文字)