package com.debuggg.java.exer2;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

public class ProxyTest {
public static void main(String[] args) {
SuperMan superMan = new SuperMan();
Human proxyInstance = (Human) ProxyFactory.getProxyInstance(superMan);
//当通过代理类对象调用方法时,会自动的调用被代理类中同名的方法
proxyInstance.eat("米饭");
System.out.println(proxyInstance.getBelief());
}
}

interface Human{
String getBelief();

void eat(String food);
}

/**
* 作者 ZYL
* 功能描述 : 被代理类
* 日期 2020-03-01 11:31
* 参数 null
* 返回值
*/
class SuperMan implements Human{

@Override
public String getBelief() {
return "I believe I can fly";
}

@Override
public void eat(String food) {
System.out.println("我喜欢吃" + food);
}
}

class HumanUtil{
public void method1(){
System.out.println("通用方法一");
}

public void method2(){
System.out.println("通用方法二");
}
}

/**
* 作者 ZYL
* 功能描述 : 动态代理
* 问题一:如何根据加载到内存中的被代理类,动态的创建一个代理类及其对象
* 问题二:当通过代理类的对象调用方法时,如何动态的去调用被代理类中的同名方法
* 日期 2020-03-01 11:34
* 参数 null
* 返回值
*/

class ProxyFactory {
/**
* 作者 ZYL
* 功能描述 : 调用此方法,返回一个代理类的对象。解决以上问题一
* 日期 2020-03-01 11:37
* 参数 object
* 返回值 java.lang.Object
*/
public static Object getProxyInstance(Object object){//object:被代理类的对象
MyInvocationHandler handler = new MyInvocationHandler();
handler.bind(object);
Object o = Proxy.newProxyInstance(object.getClass().getClassLoader(), object.getClass().getInterfaces(),handler);
return o;
}
}

class MyInvocationHandler implements InvocationHandler{
private Object obj;//赋值时也需要使用被代理类的对象进行赋值

public void bind(Object obj){
this.obj = obj;
}

/**
* 作者 ZYL
* 功能描述 : 当我们通过代理类的对象,调用方法a时,就会自动的调用如下的方法:invoke()
* 将被代理类要执行的方法a的功能,就声明在invoke中
* 日期 2020-03-01 11:40
* 参数 proxy
* 参数 method
* 参数 args
* 返回值 java.lang.Object
*/
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
HumanUtil humanUtil = new HumanUtil();

humanUtil.method1();
//method:即为代理类对象掉用的方法,此方法也就作为了被代理类对象要调用的方法
//obj:被代理类的对象
Object returnValue = method.invoke(obj, args);
//上述方法的返回值就作为当前类中的invoke()的返回值

humanUtil.method2();
return returnValue;
}
}