- 设计模式:一些通用的解决固定问题的方式
代理模式
作用:可以增强对象的功能
- 概念:
- 真实对象:被代理的对象
- 代理对象:
- 代理模式:代理对象代理真实对象,达到增强真实对象功能的目的
- 实现方式:
- 静态代理:有一个类文件描述代理模式
- 动态代理:在内存中形成代理类(本篇博客只介绍动态代理)
- 实现步骤(重点):
- 代理对象和真实对象实现相同的接口
- 代理对象 = Proxy.newProxyInstance();
- 使用代理对象调用方法。
- 增强方法
- 增强方式(重点,要不然代理就没意义,就是来增强的):
- 增强参数列表
- 增强返回值类型
- 增强方法体执行逻辑
代理模式实现步骤和增强方式见下面代码:(建一个普通java class)
使用代理对象就要在invoke写代码进行输出,创建代理对象后面的代码不是直接输出的,而是传给invoke输出的。
package cn.itcast.proxy_package;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
/**
* @author QLBF
* @version 1.0
* @date 2020/12/2 21:29
*/
//先定义一个共同的接口:
interface SaleComputer{
public String sale(double money);
public void show();
}
//定义一个真实类(真实对象),实现上面接口
class Lenovo implements SaleComputer{
@Override
public String sale(double money) {
System.out.println("花了"+money+"元买了一台联想电脑");
return "联想电脑";
}
@Override
public void show() {
System.out.println("展示电脑...");
}
}
//定义测试类,里面有代理对象
public class Proxy_test {
public static void main(String[] args) {
//1.创建真实对象
Lenovo lenovo=new Lenovo();
//2.动态代理增强lenovo对象,下面三个参数都是固定的,创建代理对象 = Proxy.newProxyInstance();
/*
三个参数:
1. 类加载器:真实对象.getClass().getClassLoader()
2. 接口数组:真实对象.getClass().getInterfaces()
3. 处理器:new InvocationHandler(),是个匿名对象,需实现invoke接口
*/
//原本是Object一般转为SaleComputer(你定义的接口类型),向下转型,返回值proxy_lenovo 就是个代理对象,用接口来接收它。
SaleComputer proxy_lenovo = (SaleComputer)Proxy.newProxyInstance(lenovo.getClass().getClassLoader(), lenovo.getClass().getInterfaces(), new InvocationHandler() {
/*
代理逻辑编写的方法:代理对象调用的所有方法都会触发该方法执行
参数:
1. proxy:代理对象
2. method:代理对象调用的方法,被封装为的对象(就把下面调用方法的String computer = proxy_lenovo.sale(8000);转到这里封装成对象的)
3. args:代理对象调用的方法时,传递的实际参数
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
/* System.out.println("该方法执行了...");
System.out.println(method.getName());//sale:看你下面调用方法的方法名
System.out.println(args[0]);//8000.0:看你下面调用方法的实际参数,锐下面调用方法没参数,这句话就不要试了,会报空指针异常
//下面两句基本固定,就是传真实类+参数,这样代理对象就可以调用真实对象的方法,会输出的了
//使用真实对象调用该方法
Object obj = method.invoke(lenovo,args);//花了8000.0元买了一台联想电脑:要这样才能执行真实对象的方法,
// args为调用方法的实际参数(是个数组,可多参),返回的obj是真实对象对应方法的返回值,若为void 则什么都不返回
return obj;*/ //obj为真实对象方法的返回值.
//下面演示如何增强方法了(重点)
// 判断是否是sale方法(看你下面想调用的是什么方法)
if (method.getName().equals("sale")){
//1.增强参数
//获取实际参数
double money=(double)args[0];//因为这里传进来的参数为double,根据实际情况来定
//改变实际参数
money=money*0.85;
System.out.println("我开车去接你...");
//改变实际参数再使用真实对象调用该方法(这就是增强参数了)
Object obj = method.invoke(lenovo, money);
System.out.println("免费送货...");
//return obj;
//2.增强返回值
return obj+"_鼠标垫";
}else{
Object obj = method.invoke(lenovo, args);
return obj;
}
}
});
//3.调用方法
String computer = proxy_lenovo.sale(8000);
System.out.println(computer);
//proxy_lenovo.show();
}
}
使用代理对象通过增强对象功能的方法可以改变真实对象的输出,而且它不需要定义实现某个接口的方法,就可以输出真实对象的方法了,而且还可以输出增强后的。