一、什么是动态代理?
现在要给eat方法增加其它功能,例如吃饭之前添加拿筷子,盛饭。
在已有的代码中插入,直接修改代码,我们叫做侵入式修改。而在一个成熟的项目中,这样做是很危险的,可能全崩啦!
此时想要增加额外的功能而又不能修改原有代码,如何去做呢?
此时我们可以找一个代理先帮我们做拿筷子和盛饭两个准备工作,等真正吃饭时,再去调用Student里的eat方法吃饭。这其实就是动态代理。
二、程序为什么需要代理?代理长什么样?
三、如何为Java创建一个代理对象
java.long.reflect.Proxy类,提供了为对象产生代理对象的方法
package com.mydynamicproxy1;
public class Test {
public static void main(String[] args) {
/*需求:外面的人想要大明星唱一首歌
* 1、获取代理的对象
* 代理对象 = ProxyUtil.createProxy(大明星的对象)
* 2.再调用代理的唱歌方法
* 代理对象.唱歌的方法("只因你太美");
*/
//1、获取代理的对象
BigStar bigstar = new BigStar("鸡哥");
Star proxy = ProxyUtil.createProxy(bigstar);
//2.再调用代理的唱歌方法
String result = proxy.sing("只因你太美");
System.out.println(result);
//3.调用跳舞的方法
proxy.dance();
}
}
package com.mydynamicproxy1;
public class BigStar implements Star {
private String name;
public BigStar(String name) {
this.name = name;
}
//唱歌
@Override
public String sing(String name){
System.out.println(this.name + "正在唱" + name);
return "谢谢";
}
//跳舞
@Override
public void dance(){
System.out.println(this.name + "正在跳舞");
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
package com.mydynamicproxy1;
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<?>[] interface,InvocationHandler h)
* 参数一:用于指定用哪个类加载器,去加载生成的代理类
* 参数二:指定接口,这些接口用于指定生成的代理长什么样,也就是有哪些方法
* 参数三:用来指定生成的代理对象要干什么事情
* */
Star star = (Star)Proxy.newProxyInstance(
ProxyUtil.class.getClassLoader(),//参数一:用于指定用哪个类加载器,去加载生成的代理类
new Class[]{Star.class},//参数二:指定接口,这些接口用于指定生成的代理长什么样,也就是有哪些方法
//参数三:用来指定生成的代理对象要干什么事情
new InvocationHandler() {
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
/*
* 参数一:代理的对象
* 参数二:要运行的方法 sing
* 参数三:调用sing方法时,传递的实参
*
* */
if("sing".equals(method.getName())){//判断要运行的方法是否在代理方法列表里(我代理的人可以干这个活,我先收费)
System.out.println("准备话筒,收钱");
}else if("dance".equals(method.getName())){
System.out.println("准备场地,收钱");
}
//去找大明星开始唱歌或跳舞
//代码的表现形式:调用大明星里面唱歌或者跳舞的方法
return method.invoke(bigstar,args);
}
}
);
return star;
}
}
package com.mydynamicproxy1;
public interface Star {
//我们可以把所有想要被代理的方法定义在接口当中
//唱歌
public abstract String sing(String name);
//跳舞
public abstract void dance();
}
接下来要理清调用过程(个人理解梳理)
1.首先创建被代理对象的代理对象
2.使用创建的代理对象调用方法时底层会调用invoke,将要执行的方法传给invoke的第二个参数method,要执行的方法的参数传给invoke的第三个参数args。
3.在invoke中,判断传入的要运行的方法是否在代理列表内,若是,对其进行方法的加强,如例,为大明星唱歌做准备工作。
4.接下来,调用被代理对象的方法,method就是实际的方法,args就是实际方法的形参,此时实际方法就被调用了,执行实际方法,将实际方法的返回值返回。