动态代理到基于动态代理的AOP
动态代理,是java支持的一种程序设计方法。
动态代理实现中有两个重要的接口和类,分别是InvocationHandler(interface),Proxy(class).
要实现动态代理,必须要定义一个被代理类的接口Interface,另外,被代理类必须实现这个接口,这样,在代理类生存过程中,就可以通过接口,反射的方式找到这个被代理类。这个和代理模式中要求基本一致:代理类和被代理类都必须实现同一个接口。只是这里,有点点不同,只是要求被代理类必须定义一个接口,而代理类依据被代理类接口通过反射的机制找到这个被代理类的实现。有点绕,静下心理一理,还是比较容易弄清的!
下面就用一个动态代理的例子来呈现其原理吧。
1.首先是被代理类的接口定义。
- package dynamic_proxy;
- /**
- * 动态代理,必须有一个公共的接口,这个也是代理模式的基本要求
- *
- * @author shihuc
- * @date Mar 17, 2016
- *
- */
- public interface Happy {
- public void sayHello();
- public void makeFriends(String guNiang);
- }
2.被代理类的具体实现
- package dynamic_proxy;
- /**
- *
- * 被代理的类,必须实现公共的接口,要有自己的特色活要干,比如这里,西门庆后面要被王婆代理,
- * 但是西门庆最终还是要上阵干活,泡潘金莲啊。只是外表上,让人看起来是王婆在作媒人。。。
- *
- * @author shihuc
- * @date Mar 17, 2016
- *
- */
- public class Ximenqing implements Happy{
- public void sayHello() {
- System.out.println("Hi, I am Ximenqing");
- }
- public void makeFriends(String guNiang) {
- System.out.println("Hello, " + guNiang + ", I am Ximenqing!");
- }
- }
3. 代理类的实现。这里很重要,代理类必须实现InvocationHandler这个Interface,并且实现里面的invoke函数。
- package dynamic_proxy;
- import java.lang.reflect.InvocationHandler;
- import java.lang.reflect.Method;
- /**
- *
- * 王婆开始代理西门庆上场了,注意,这里只是表面上看是王婆代理,但是,王婆其实只是牵线搭桥,比如,
- * 将Happy这个接口的实例ximenqing传进代理,后面干活的,还是这个西门庆哟!
- *
- * @author shihuc
- * @date Mar 17, 2016
- *
- */
- public class PaoniuHandler implements InvocationHandler{
- Happy happy;
- public PaoniuHandler(Happy happy){
- this.happy = happy;
- }
- public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
- //在这里,你可以加入真实函数执行前的辅助逻辑。
- System.out.println("Here, ‘BEFORE’ you can add your supplemental logic");
- //反射的机制实现对真是的被代理的函数进行执行操作
- method.invoke(happy, args);
- //在这里,你可以加入真是逻辑执行完成后的一些处理逻辑
- System.out.println("Here, 'AFTER' you can add some logic for you completed real logic");
- return null;
- }
- }
4. 下面用一个小测试程序,检测一下上面的动态代理的执行过程。
- package dynamic_proxy;
- import java.lang.reflect.InvocationHandler;
- import java.lang.reflect.Proxy;
- /**
- * 动态代理的具体运行步骤。重点是通过Proxy类来生成代理类的实例wangpo。这里的生成过程,和代理模式不同,这里是依据Proxy类的类函数newProxyInstance得到的。
- * static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h)
- * JDK1.6对上面这个函数的解释:“返回一个指定接口的代理类实例,该接口可以将方法调用指派到指定的调用处理程序。”
- *
- * @author shihuc
- * @date Mar 18, 2016
- *
- */
- public class Client {
- public static void main(String args[]){
- //真正干活的人,西门庆的诞生
- Happy ximenqing = new Ximenqing();
- //创建代理类处理程序,在这里,需要传递进被代理者,也就是泡妞的主体西门庆。
- InvocationHandler xmqHandler = new PaoniuHandler(ximenqing);
- /*代理者wangpo出场的核心程序。
- * Proxy类的newProxyInstance通过用户指定的类加载器xmqHandler.getClass().getClassLoader(),
- * 代理类要实现的接口列表ximenqing.getClass().getInterfaces(),
- * 代理类指派方法调用的调用处理程序(也就是说,代理类基于Happy接口生成代理类,这个代理类要做的事情是InvocationHandler实现者里指定被代理者做的事情。
- * 说白了,就是王婆帮西门庆出去搭讪潘金莲了,但是核心婆妞的事情还是西门庆上。)
- */
- Happy wangpo = (Happy)Proxy.newProxyInstance(xmqHandler.getClass().getClassLoader(), ximenqing.getClass().getInterfaces(), xmqHandler);
- //王婆搭讪潘金莲了。其实后手是西门庆。
- wangpo.makeFriends("Ms Pan Jinlian");
- }
- }
下面看看执行的输出:
- Here, ‘BEFORE’ you can add your supplemental logic
- Hello, Ms Pan Jinlian, I am Ximenqing!
- Here, 'AFTER' you can add some logic for you completed real logic
到这里,动态代理的实现全部完成。是不是没有想象的那么复杂,关于核心InvocationHandler以及Proxy的newProxyInstance的具体实现,可以google,很多的!
接下来,看看如何基于动态代理,实现基于动态代理的AOP。其实改动的地方很小,下面给一个简单的例子,基于上面的例子做了点改动。
- package dyn_pxy_aop;
- /**
- * 动态代理,必须有一个公共的接口,这个也是代理模式的基本要求
- *
- * @author shihuc
- * @date Mar 17, 2016
- *
- */
- public interface Happy {
- public void sayHello();
- public void makeFriends(String guNiang);
- }
- package dyn_pxy_aop;
- /**
- *
- * 被代理的类,必须实现公共的接口,要有自己的特色活要干,比如这里,西门庆后面要被王婆代理,
- * 但是西门庆最终还是要上阵干活,泡潘金莲啊。只是外表上,让人看起来是王婆在作媒人。。。
- *
- * @author shihuc
- * @date Mar 17, 2016
- *
- */
- public class Ximenqing implements Happy{
- public void sayHello() {
- System.out.println("Hi, I am Ximenqing");
- }
- public void makeFriends(String guNiang) {
- System.out.println("Hello, " + guNiang + ", I am Ximenqing!");
- }
- }
- package dyn_pxy_aop;
- import java.lang.reflect.InvocationHandler;
- import java.lang.reflect.Method;
- /**
- *
- * 王婆开始代理西门庆上场了,注意,这里只是表面上看是王婆代理,但是,王婆其实只是牵线搭桥,比如,
- * 将Happy这个接口的实例ximenqing传进代理,后面干活的,还是这个西门庆哟!
- *
- * @author shihuc
- * @date Mar 17, 2016
- *
- */
- public class PaoniuHandler implements InvocationHandler{
- Happy happy;
- public PaoniuHandler(Happy happy){
- this.happy = happy;
- }
- public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
- //反射的机制实现对真是的被代理的函数进行执行操作
- method.invoke(happy, args);
- return null;
- }
- }
- package dyn_pxy_aop;
- import java.lang.reflect.InvocationHandler;
- import java.lang.reflect.Method;
- /**
- * 王婆的代理,其产生的机制,就非常的明显了,基于反射获取Happy的方法,然后直接利用InvocationHandler的invoke方法进行方法调用,
- * 更进一步的显示出真正干活的是谁了(西门庆)
- *
- * @author shihuc
- * @date Mar 18, 2016
- *
- */
- public class WangpoProxy implements Happy{
- InvocationHandler paoniuHandler;
- public WangpoProxy(InvocationHandler h){
- this.paoniuHandler = h;
- }
- public void sayHello() {
- try {
- Method m = ((PaoniuHandler)paoniuHandler).happy.getClass().getMethod("sayHello", null);
- paoniuHandler.invoke(this, m, null);
- } catch (NoSuchMethodException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- } catch (SecurityException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- } catch (Throwable e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- public void makeFriends(String guNiang) {
- try {
- Method m = ((PaoniuHandler)paoniuHandler).happy.getClass().getMethod("makeFriends", new Class[]{String.class});
- paoniuHandler.invoke(this, m, new Object[]{guNiang});
- } catch (NoSuchMethodException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- } catch (SecurityException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- } catch (Throwable e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- }
- package dyn_pxy_aop;
- /**
- * 所谓的AOP,最终的目的就是在不修改原有的业务代码的基础上,在原有逻辑的前(before),后(after),以及前和后(around)的位置,添加其他的功能(advice),比如日志,事物,安全等
- *
- * @author shihuc
- * @date Mar 18, 2016
- *
- */
- public class AOP_demo {
- public static void main(String[] args) {
- PaoniuHandler ph = new PaoniuHandler(new Ximenqing());
- //这里可以加入advice
- new WangpoProxy(ph).sayHello();
- new WangpoProxy(ph).makeFriends("Pan Jinlian");
- //这里可以加入advice
- }
- }
基于动态代理实现的AOP,其实有个很明显的不足,就是过于依赖反射,这就必然影响性能。但是它也有个明显的优势,就是实现起来灵活。
AOP的实现,除了基于动态代理的方案,还有动态字节码生成方案,静态AOP等等,可以慢慢研究,根据需要进行取舍!
动态代理到基于动态代理的AOP的更多相关文章
- Web静态和动态项目委托代理基于面向方面编程AOP
本来每天更新,我一般喜欢晚上十二点的时候发文章,结果是不是愚人节?校内网也将是非常有趣,破,把我给打. ..好吧-从今天开始的话题AOP.AOP太重要了,所以把第二篇文章谈论这个话题,AOP它是Spr ...
- 最简单的动态代理实例(spring基于接口代理的AOP原理)
JDK的动态代理是基于接口的 package com.open.aop; public interface BusinessInterface { public void processBus ...
- Spring AOP源码分析(三):基于JDK动态代理和CGLIB创建代理对象的实现原理
AOP代理对象的创建 AOP相关的代理对象的创建主要在applyBeanPostProcessorsBeforeInstantiation方法实现: protected Object applyBea ...
- .NET 下基于动态代理的 AOP 框架实现揭秘
.NET 下基于动态代理的 AOP 框架实现揭秘 Intro 之前基于 Roslyn 实现了一个简单的条件解析引擎,想了解的可以看这篇文章 https://www.cnblogs.com/weihan ...
- 实现一个简单的基于动态代理的 AOP
实现一个简单的基于动态代理的 AOP Intro 上次看基于动态代理的 AOP 框架实现,立了一个 Flag, 自己写一个简单的 AOP 实现示例,今天过来填坑了 目前的实现是基于 Emit 来做的, ...
- Spring AOP详解 、 JDK动态代理、CGLib动态代理
AOP是Aspect Oriented Programing的简称,面向切面编程.AOP适合于那些具有横切逻辑的应用:如性能监测,访问控制,事务管理以及日志记录.AOP将这些分散在各个业务逻辑中的代码 ...
- 何为代理?jdk动态代理与cglib代理、spring Aop代理原理浅析
原创声明:本博客来源为本人原创作品,绝非他处摘取,转摘请联系博主 代理(proxy)的定义:为某对象提供代理服务,拥有操作代理对象的功能,在某些情况下,当客户不想或者不能直接引用另一个对象,而代理对象 ...
- 【转载】Spring AOP详解 、 JDK动态代理、CGLib动态代理
Spring AOP详解 . JDK动态代理.CGLib动态代理 原文地址:https://www.cnblogs.com/kukudelaomao/p/5897893.html AOP是Aspec ...
- 基于JDK动态代理和CGLIB动态代理的实现Spring注解管理事务(@Trasactional)到底有什么区别。
基于JDK动态代理和CGLIB动态代理的实现Spring注解管理事务(@Trasactional)到底有什么区别. 我还是喜欢基于Schema风格的Spring事务管理,但也有很多人在用基于@Tras ...
随机推荐
- 原来DataTable的Distinct竟如此简单!
DataView可以帮我们直接获取Distinct数据, DataTable dataTable;DataView dataView = dataTable.DefaultView; DataTabl ...
- Warning: Data truncated for column 'AirPress' at row 1
/************************************************************************ * Warning: Data truncated ...
- VMware虚拟机固定IP后克隆出现无法访问网卡问题
通常我们现在都喜欢使用虚拟机进行实验,进行集群搭建等,在这个过程中,会遇到克隆虚拟机问题,当没有修改任何IP的情况下,克隆后,在逐台修改IP地址是没有问题的,但是,如果我们先设置了固定IP地址后,克隆 ...
- ARM处理器模式
ARM处理器模式 ARM处理器共有7种运行模式: l 用户模式(User,usr):正常程序执行的模式 l 快速中断模式(FIQ,fiq):用于高速数据传输和通道处理 l 外部中断模式(IRQ, ...
- python 中 sorted() 和 list.sort() 的用法
今天用python自带的sorted对一个列表进行排序, 在这里总结一下 只要是可迭代对象都可以用sorted . sorted(itrearble, cmp=None, key=None, reve ...
- jq中如何阻止元素的默认行为?
阻止网页元素的默认行为: event.preventDefault(); 或者:return false;
- 网站优化之-SEO在网页制作中的应用(信息来自慕课网课程笔记)
一.SEO基本介绍. 1.搜索引擎工作原理. 2.seo简介:SEarch Engine Optimization,搜索引擎优化.为了提升网页在搜索引擎自然搜索结果中的收录数量及排序位置而做的优化行为 ...
- PHP 页面自动刷新可借助JS来实现,简单示例如下:
<?php echo "系统当前时间戳为:"; echo ""; echo time(); //<!--JS 页面自动刷新 --> echo ...
- ajax请求后台,返回json格式数据,模板!
添加一个用户的时候,需要找出公司下所有的部门,和相应部门下的角色,利用ajax请求,实现联动技术.将返回的json格式数据,添加到select标签下. <script type="te ...
- 机器学习&人工智能书籍
Introduction to Machine Learning https://www.amazon.cn/Introduction-to-Machine-Learning-Alpaydin-Eth ...