什么是动态代理?
例子:
明星会唱歌,跳舞。但是开演唱会如果还兼顾其他事务工作的话,事情就会太多,因此这时候就可以使用代理,转移部分事情。
(代理会先把准备工作做完,然后再去调用对象中的方法)
⚪对象如果嫌身上干的事太多的话,可以通过代理来转移部分职责。
⚪对象有什么方法想被代理,代理就一定要有对应的方法.
1.为什么需要代理?
代理可以无侵入式的给对象增强其他的功能
调用者–>代理–>对象
2.代理长什么样?
代理的里面就是对象要被代理的方法
3.Java通过上面来保证代理的样子?
通过接口保证,后面的对象和代理需要实现同一个接口
接口中就是被代理的所有方法
如何为Java对象创建一个代理对象
java.lang.reflect.Proxy类型:提供了为对象产生代理对象的方法:
public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)
参数一:用于指定用哪个类加载器,去加载生成的代理类
参数二:指定接口,这些接口用于指定生成的代理长什么,也就是有哪些方法。
参数三:用来指定生成的代理对象要干上面事情。
案例:
明星类:BigStar
package Xia;
public class BigStar implements Star {
private String name;
public BigStar() {
}
public BigStar(String name) {
this.name = name;
}
//唱歌,跳舞
public String sing(String geming) {
System.out.println(this.name + "正在唱" + geming);
return "谢谢";
}
public void dance() {
System.out.println(this.name + "正在跳舞");
}
/**
* 获取
*
* @return name
*/
public String getName() {
return name;
}
/**
* 设置
*
* @param name
*/
public void setName(String name) {
this.name = name;
}
public String toString() {
return "BigStar{name = " + name + "}";
}
}
接口:Star 里面是想要被代理的所有方法
package Xia;
public interface Star {
// 我们可以把所有想要被代理的方法定义再接口中
//唱歌
public abstract String sing(String geming);
//跳舞
public abstract void dance();
}
代理类 Proxy :创建一个代理对象,重中之重
package Xia;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
//类的作用: 创建一个代理
public class ProxyUtil {
/*
*
* 方法作用: 给一个明星对象,创建一个代理
*
* 形参:被代理的明星对象
*
* 返回值:给明星创建的代理
*
* 需求: 外面的人想要大明星唱一首歌
* 1,获取代理的对象
* 代理对象 = ProxyUtil.createProxy(大明星对象);
*
* 2.再调用代理的唱歌方法
* 代理对象.唱歌的方法(”及你太美“);
*
* */
public static Star createProxy(BigStar bigStar) {
/*
* java.lang.reflect.Proxy类型:提供了为对象产生代理对象的方法:
public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)
参数一:用于指定用哪个类加载器,去加载生成的代理类
参数二:指定接口,这些接口用于指定生成的代理长什么,也就是有哪些方法。
参数三:用来指定生成的代理对象要干上面事情。*/
Star star = (Star) Proxy.newProxyInstance(
//第一个获取这个类的class对象,然后class对象调用方法寻找这个类加载进行内存的加载器。 可以理解为::找个人给自己送进去
ProxyUtil.class.getClassLoader(),
//可以理解为: 中介可以代理哪些方法,如果有多个方法都可以写在数组中
new Class[]{Star.class}, // 代理和代理对象都需要实现的接口方法
//
new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { //代理实现方法是真正执行的流程
//参数一: 代理的对象
//参数二: 要运行的方法 sing
//参数三: 调用方法是要传递的 参数
if ("sing".equals(method.getName())) {
System.out.println("准备话筒,收钱");
}
if ("dance".equals(method.getName())) {
System.out.println("准备场地,收钱");
}
//去找大明星开始唱歌跳舞
//代码的表现形式:调用大明星里面唱歌或者跳舞的方法
return method.invoke(bigStar, args);
}
});
return star;
}
}
实现代理
package Xia;
public class DaiLiTest {
public static void main(String[] args) {
/*
* 需求: 外面的人想要大明星唱一首歌
* 1,获取代理的对象
* 代理对象 = ProxyUtil.createProxy(大明星对象);
* 2.再调用代理的唱歌方法
* 代理对象.唱歌的方法(”及你太美“);
*/
//1 . 获取代理对象
BigStar bigStar = new BigStar("鸡哥"); //被代理的明星对象
Star proxy = ProxyUtil.createProxy(bigStar); // 给鸡哥生成了一个代理对象
//调用唱歌的方法
String result = proxy.sing("及你太美"); // proxy会自动调用invoke方法,会将sing 和 及你太美 传入进去。
System.out.println(result);
//调用跳舞的方法
proxy.dance();
}
}