北京网站建设外包公司排名,周浦做网站公司,甘肃网站建设公司,wordpress适配大家好#xff01;我是CSRobot#xff0c;从今天开始#xff0c;我将会发布一些技术文章#xff0c;内容就是结合春招以来的面试所遇到的问题进行分享#xff0c;首先会对知识点进行一个探讨和整理#xff0c;在最后会给出一些面试题并作出解答#xff0c;希望可以帮助到…大家好我是CSRobot从今天开始我将会发布一些技术文章内容就是结合春招以来的面试所遇到的问题进行分享首先会对知识点进行一个探讨和整理在最后会给出一些面试题并作出解答希望可以帮助到大家今天知识点是动态代理会分从以下几个方面进行探讨静态代理JDK动态代理Cglib动态代理面试题分享在java中要说到动态代理那么首先需要聊聊静态代理一、静态代理静态代理是代理类在编译期间就创建好了不是编译器生成的代理类而是手动创建的类。在编译时就已经将接口被代理类代理类等确定下来。以下以一个例子来说明静态代理的实现1、实现步骤创建服务类接口BuyHouse//创建服务接口
public interface BuyHouse {void buyHouse();
}实现服务接口 BuyHouseImpl //实现接口
public class BuyHouseImpl implements BuyHouse {Overridepublic void buyHouse() {System.out.println(我要买房子);}
}创建代理类实现服务接口在代理类中手动增强 BuyHouseProxy public class BuyHouseProxy implements BuyHouse {private BuyHouse buyHouse;public BuyHouseProxy(final BuyHouse buyHouse) {this.buyHouse buyHouse;}//手动增强buyHouse方法Overridepublic void buyHouse() {System.out.println(买房前准备);buyHouse.buyHouse();System.out.println(买房后装修);}
}创建测试类Test
public void test() {//未增强的类BuyHouse buyHouse new BuyHouseImpl();buyHouse.buyHouse();//创建代理类BuyHouseProxy buyHouseProxy new BuyHouseProxy(buyHouse);buyHouseProxy.buyHouse();
}输出我要买房子 买房前准备 我要买房子 买房后装修 二、动态代理 Java动态代理的优势是实现无侵入式的代码扩展也就是方法的增强让你可以在不用修改源码的情况下增强一些方法在方法的前后你可以做你任何想做的事情。 在java的动态代理中主要有两种实现方式首先是基于反射的JDK动态代理实现其次是cglib的字节码技术的实现1、JDK动态代理通过实现 InvocationHandler 接口创建自己的调用处理器通过为 Proxy 类指定 ClassLoader 对象和一组 interface 来创建动态代理类通过反射机制获得动态代理类的构造函数其唯一参数类型是调用处理器接口类型通过构造函数创建动态代理类实例构造时调用处理器对象作为参数被传入代理类名称是包名$Proxyid序号 JDK动态代理只能代理接口而不能代理类原因大家可以根据jdk动态代理的实现步骤大胆推断一下后文会给出答案 创建服务接口//创建服务接口
public interface BuyHouse {void buyHouse();
}public class BuyHouseImpl implements BuyHouse {Overridepublic void buyHouse() {System.out.println(我要买房子);}
}编写动态处理器public class DynamicProxyHandler implements InvocationHandler{private Object object;public DynamicProxyHandler(final Object object) {this.object object;}//invoke方法对业务进行增强Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println(买房前准备);Object result method.invoke(object, args);System.out.println(买房后装修);return result;}public Object getProxyInstance(){return Proxy.newProxyInstance(object.getClass().getClassLoader(), object.getClass().getInterfaces(), this);}
}测试动态代理Testpublic void test1() {//创建需要被代理的类BuyHouse buyHouse new BuyHouseImpl();//创建动态代理处理器DynamicProxyHandler DP new DynamicProxyHandler(buyHouse);//获得代理类BuyHouse proxyBuyHouse DP.getProxyInstance();proxyBuyHouse.buyHouse();}输出买房前准备 我要买房子 买房后装修 2、Cglib动态代理在cglib中提供了一个Enhance类来创建代理类类似于jdk动态代理中的Proxy类。添加maven依赖包dependencygroupIdcglib/groupIdartifactIdcglib/artifactIdversion2.2.2/version
/dependency同样是使用上文中buyHouse的类//创建服务接口
public interface BuyHouse {void buyHouse();
}public class BuyHouseImpl implements BuyHouse {Overridepublic void buyHouse() {System.out.println(我要买房子);}
}创建方法拦截器public class CglibProxy implements MethodInterceptor{private Object target;public Object getInstance(final Object target) {this.target target;Enhancer enhancer new Enhancer();enhancer.setSuperclass(this.target.getClass());enhancer.setCallback(this);return enhancer.create();}Overridepublic Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {System.out.println(买房前准备);Object result proxy.invokeSuper(obj, args);System.out.println(买房后装修);return result;}
}测试Test
public void test3(){BuyHouse buyHouse new BuyHouseImpl();CglibProxy cglibProxy new CglibProxy();BuyHouseImpl buyHouseCglibProxy (BuyHouseImpl)cglibProxy.getInstance(buyHouse);buyHouseCglibProxy.buyHouse();
}最终输出结果和之前相同三、面试题1、动态代理是什么动态代理为其他对象提供一个代理以控制对某个对象的访问。代理类主要负责为委托了真实对象预处理消息、过滤消息、传递消息给委托类代理类不现实具体服务而是利用委托类来完成服务并将执行结果封装处理2、为什么不使用静态代理而使用动态代理通过前文大家也可以理解了静态代理是代理类在编译期间就创建好了不是编译器生成的代理类需要手动创建的类。在编译时就已经将接口被代理类代理类等确定下来。如果我们有很多类很多接口需要代理那么我们就只能使用代码提前写死不够灵活使用了动态代理之后我们不需要手动创建代理类全部交给代理去完成对代理类的创建实现无侵入式的代码扩展这也是符合面向对象编程原则的操作。3、JDK动态代理为什么只能代理接口而不能代理类通过JDK动态代理实现步骤我们就可以看到我们通过Proxy类的newProxyInstance方法来生成代理对象代理类继承了Proxy类并且实现了要代理的接口由于java不支持多继承所以JDK动态代理不能代理类4、JDK动态代理和CgLib动态代理的区别1JDK动态代理只能代理接口而不能代理类CgLib对于接口和类都可以实现代理2JDK动态代理底层使用反射的方式实现而CgLib采用了非常底层的字节码技术其原理是通过目标类的字节码为一个类创建子类并在子类中采用方法拦截的技术拦截所有父类方法的调用顺势织入横切逻辑。代理类将目标类作为自己的父类并为其中的每个非final委托方法创建两个方法CSRobot