`
bolide74
  • 浏览: 85614 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

Spring温故知新(六)AOP面向切面编程 <1>

阅读更多
这一章我们将开始剖析Spring框架最为重要的AOP(Aspect Oriented Programming)面向切面编程。可以说Spring的精华就在于AOP了。
所谓AOP,就是相对于OOP(Object Oriented Programming)面向对象编程的说法,有些人喜欢叫面向切面编程,有些人喜欢叫做面向方面,事实上这两个都是指同一个东西,只是叫法不同。
我们传统的编程都是面向对象,就是说每个类都有它实际的意义。而面向切面略有不同,它在面向对象的基础上扩展了一下,它编程的时候不是先考虑的一个具体对象(比如用户类),而是先考虑的对象的行为或者功能。这个不是编程方法的不同,而是编程思维的转变。

理论性的东西还是放一边,我们用实际的机器人案例来慢慢理解这个概念。
为了突出重点我们这里重写了ISpeak:
package com.iteye.bolide74.impl;

public interface ISpeaker {
	public void say(String msg);
}

接着是实现这个接口的机器人类:
package com.iteye.bolide74.action;

import com.iteye.bolide74.impl.ISpeaker;

public class Robot implements ISpeaker {
	public String name;

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public Robot(String name) {
		this.name = name;
	}

	@Override
	public void say(String msg) {
		System.out.println("到达邻居家,对邻居说:" + msg + ",我是" + this.name);
	}
}

那么现在我们要实现跟很多个不同的邻居(不同的应用代码)打招呼,只需要下面的应用代码了:
package com.iteye.bolide74.tester;

import com.iteye.bolide74.action.Robot;
import com.iteye.bolide74.impl.ISpeaker;

public class Tester {
	public static void main(String[] args) {
		ISpeaker speaker = new Robot("Wall-E");
		speaker.say("你好");
	}
}


以上是实现最简单功能的方法。现在我们要慢慢增加功能了,假设Wall-E在出发打招呼之前要先拿一个礼物,然后打完招呼以后把礼物递给邻居,该怎么实现呢?
	public void getGift() {
		System.out.println("获取了一个礼物");
	}
	public void giveGift(){
		System.out.println("赠予一个礼物");
	}


最傻的办法就是每次在应用代码里调用speaker.say()方法的前后都调用get/give方法,小程序还好,要是大型程序的话人都要写傻掉。
那么能不能直接把get/give方法写在Robot类的say()方法里呢?这样好处是每次调用say()的时候都能实现get/give功能,但是坏处也是这个:万一有个别邻居是不需要给礼物的呢?怎么办?
那有没有更加灵活的办法呢?当然有!

代理模式:
我们先新建一个SpeakerProxy代理类:
package com.iteye.bolide74.action;

import com.iteye.bolide74.impl.ISpeaker;

public class SpeakerProxy implements ISpeaker {
	ISpeaker speaker;

	public SpeakerProxy(ISpeaker speaker) {
		super();
		this.speaker = speaker;
	}

	@Override
	public void say(String msg) {
		getGift();
		speaker.say(msg);
		giveGift();
	}

	public void getGift() {
		System.out.println("获取了一个礼物");
	}
	public void giveGift(){
		System.out.println("赠予一个礼物");
	}
}

这个SpeakerProxy类实现了ISpeaker接口,然后又有一个靠构造函数传入的ISpeaker类型的成员属性。
而这个SpeakerProxy类的say()方法就有点意思了,它不是重写了自己的say()方法,而是调用了ISpeaker类型的speaker成员属性的say()方法,然后再这个say()方法前后嵌入get/give方法。
让我们看看这个SpeakerProxy类是怎么在应用代码使用:
package com.iteye.bolide74.tester;

import com.iteye.bolide74.action.Robot;
import com.iteye.bolide74.action.SpeakerProxy;
import com.iteye.bolide74.impl.ISpeaker;

public class SpeakerProxyTester {
	public static void main(String[] arg0) {
		// 没有带礼物的机器人:
		ISpeaker noGiftSpeaker = new Robot("空手来的Wall-E");
		noGiftSpeaker.say("你好");
		System.out.println();
		// 带了礼物的机器人
		ISpeaker speaker = new SpeakerProxy(new Robot("有礼而来的Wall-E"));
		speaker.say("你好");
	}
}

输出结果:
引用
到达邻居家,对邻居说:你好,我是空手来的Wall-E

获取了一个礼物
到达邻居家,对邻居说:你好,我是有礼而来的Wall-E
赠予一个礼物


这样的话我们就达到了灵活性:不需要礼物的时候直接新建一个机器人;而需要礼物的时候我们就new一个SpeakerProxy(代理器),然后新建一个机器人丢进去,我们只要调用这个代理器就行了,这个代理器内部会给机器人一个礼物然后命令机器人去打招呼送礼物。


扩展阅读:装饰模式:
引用
        装饰模式与AOP关系不大,但是它的实现方法与代理模式很像,不乘此机会一起介绍有点可惜,如果已经了解了装饰模式的可以直接跳过。

假设我们的机器人要送的礼物种类比较多,有红包、水果、花等等。而给不同的邻居打招呼送礼的时候都会送不同组合的礼物,比如:红包+水果、红包+花、水果+花、全部都送、单独送其中一样。
那么这种功能该怎么实现呢?每种组合写一个get/give方法吗?如果只有简单3种组合可能忍忍也就算了,但是如果以后礼物种类越来越多,那这个组合数量是成爆发性增长的,显然不合适。
那么这时候就该轮到装饰模式登场了:
package com.iteye.bolide74.action;

import com.iteye.bolide74.impl.ISpeaker;

public abstract class SpeakerGiftDecorator implements ISpeaker {
	ISpeaker speaker;

	public SpeakerGiftDecorator(ISpeaker speaker) {
		super();
		this.speaker = speaker;
	}

	public abstract void say(String msg);

	public abstract void getGift();

	public abstract void giveGift();
}

这是一个礼物的装饰器类,你可以理解为各种礼物的组装器。它的写法跟代理模式的代理类很像,不同的就是它是一个抽象类。它是所有种类的礼物的父类。
接下来就写一个继承了这个SpeakerGiftDecorator类的礼物之类:
package com.iteye.bolide74.action;

import com.iteye.bolide74.impl.ISpeaker;
//礼物的一种:一束花
public class Flower extends SpeakerGiftDecorator {

	public Flower(ISpeaker speaker) {
		super(speaker);
	}

	@Override
	public void say(String msg) {
		getGift();
		this.speaker.say(msg);
		giveGift();
	}

	@Override
	public void getGift() {
		System.out.println("获取了一束花");
	}

	@Override
	public void giveGift() {
		System.out.println("赠予一束花");
	}

}

以上面的Flower子类类推来写出另外的Money类和Fruit类,区别仅在于get/give方法的内容不同,这里就不浪费篇幅了。

看起来代码似乎很简单,没什么出奇的,但是当应用代码使用的时候就能体现出它强大的地方了:
package com.iteye.bolide74.tester;

import com.iteye.bolide74.action.Flower;
import com.iteye.bolide74.action.Fruit;
import com.iteye.bolide74.action.Money;
import com.iteye.bolide74.action.Robot;
import com.iteye.bolide74.impl.ISpeaker;

public class SpeakerGiftDecoratorTester {
	public static void main(String[] args) {
		// 一种礼物:
		ISpeaker speaker = new Flower(new Robot("我是带花来的Wall-E"));
		speaker.say("Hello");
		System.out.println();
		// 两种礼物:
		speaker = new Money(new Flower(new Robot("我是带了花和红包的Wall-E")));
		speaker.say("Hello");
		System.out.println();
		// 另外两种礼物:
		speaker = new Fruit(new Money(new Robot("我是带了水果和钱的Wall-E")));
		speaker.say("Hello");
		System.out.println();
		// 三种礼物:
		speaker = new Fruit(new Money(new Flower(new Robot("我是三种礼物都带的Wall-E"))));
		speaker.say("Hello");
	}
}

可以看到,speaker引用的是一个一层一层嵌套的类,要多带一种礼物那就多嵌套一层,这样就能尽可能的实现了代码的重用,而且应用起来也简单明了。


输出结果:
引用
获取了一束花
到达邻居家,对邻居说:Hello,我是我是带花来的Wall-E
赠予一束花

获取了一个红包
获取了一束花
到达邻居家,对邻居说:Hello,我是我是带了花和红包的Wall-E
赠予一束花
赠予一个红包

获取了一袋水果
获取了一个红包
到达邻居家,对邻居说:Hello,我是我是带了水果和钱的Wall-E
赠予一个红包
赠予一袋水果

获取了一袋水果
获取了一个红包
获取了一束花
到达邻居家,对邻居说:Hello,我是我是三种礼物都带的Wall-E
赠予一束花
赠予一个红包
赠予一袋水果



我的围脖:http://t.qq.com/bolide74


下一篇:Spring温故知新(六)AOP面向切面编程 <2> http://bolide74.iteye.com/blog/1007828
上一篇:Spring温故知新(五)Spring的Bean和IoC 容器 http://bolide74.iteye.com/blog/1004086








10
4
分享到:
评论
3 楼 chenming47 2011-04-25  
不错,讲的通俗易懂
2 楼 qj200040 2011-04-24  
嗯,不错的例子,我要好好学习学习,谢谢你哈
1 楼 步青龙 2011-04-19  
  不错,这样的例子很有感觉

相关推荐

    Spring温故知新六AOP向切面程

    Spring框架的核心特性之一就是AOP(Aspect Oriented Programming,面向切面编程)。AOP提供了一种模块化和声明式的方式来处理系统中的横切关注点,如日志、事务管理、权限检查等。在OOP(面向对象编程)中,这些关注...

    spring_2021.7.14.zip

    3. "spring01quickstart.zip":这是Spring框架的基础快速启动教程,可能涉及到Spring的核心概念,如IoC容器、bean的生命周期、AOP(面向切面编程)等。这部分可能会介绍如何创建第一个Spring应用,包括引入Spring库...

    Spring全家桶思维导图

    首先,Spring的核心组件包括Spring Core和Spring Beans,它们提供了依赖注入(DI)和面向切面编程(AOP)的基础。依赖注入使得对象之间的关系在运行时动态配置,降低了代码间的耦合度;面向切面编程则允许我们分离...

    电子工程0欧姆电阻在PCB设计中的多功能应用

    内容概要:0欧姆电阻在电路设计中有多种重要作用。它不仅可以在PCB上为调试提供便利,还能用于跳线、替代不确定参数的元件以及测量电路的耗电流。此外,在布线困难时可作为应急解决方案。在高频信号环境下,它能充当电感或电容,有助于解决EMC问题。对于地线处理,0欧姆电阻可用于实现单点接地,避免模拟地和数字地直接大面积相连带来的互相干扰问题。在跨接电流回路方面,它可以提供较短的回流路径,减少干扰。同时,0欧姆电阻还适用于配置电路,防止用户误操作跳线或拨码开关,并且在布线、调试、测试、温度补偿等方面有着广泛应用,尤其在EMC对策中表现突出。; 适合人群:电子工程师、硬件设计师以及对电路设计感兴趣的爱好者。; 使用场景及目标:①在PCB设计阶段,利用0欧姆电阻进行灵活的电路调试与优化;②解决高频信号下的EMC问题,确保电路稳定性和抗干扰能力;③实现单点接地,避免不同地线间的相互干扰;④提高电路的可维护性和可靠性,降低生产成本。; 阅读建议:本文详细介绍了0欧姆电阻在电路设计中的多种应用场景,读者应结合具体项目需求来理解和运用这些知识,特别是在面对复杂的电路布局和电磁兼容性问题时,要充分考虑0欧姆电阻的独特优势。

    一个基于SpringBoot+Mybatis+Mysql+Html实现的页面登录案例

    mysql安装教程 一个基于SpringBoot+Mybatis+Mysql+Html实现的页面登录案例.

    全域旅游综合解决方案PPT(71页).pptx

    在探索智慧旅游的新纪元中,一个集科技、创新与服务于一体的整体解决方案正悄然改变着我们的旅行方式。智慧旅游,作为智慧城市的重要分支,旨在通过新一代信息技术,如云计算、大数据、物联网等,为游客、旅游企业及政府部门提供无缝对接、高效互动的旅游体验与管理模式。这一方案不仅重新定义了旅游行业的服务标准,更开启了旅游业数字化转型的新篇章。 智慧旅游的核心在于“以人为本”,它不仅仅关注技术的革新,更注重游客体验的提升。从游前的行程规划、信息查询,到游中的智能导航、个性化导览,再到游后的心情分享、服务评价,智慧旅游通过构建“一云多屏”的服务平台,让游客在旅游的全过程中都能享受到便捷、个性化的服务。例如,游客可以通过手机APP轻松定制专属行程,利用智能语音导览深入了解景点背后的故事,甚至通过三维GIS地图实现虚拟漫游,提前感受目的地的魅力。这些创新服务不仅增强了游客的参与感和满意度,也让旅游变得更加智能化、趣味化。 此外,智慧旅游还为旅游企业和政府部门带来了前所未有的管理变革。通过大数据分析,旅游企业能够精准把握市场动态,实现旅游产品的精准营销和个性化推荐,从而提升市场竞争力。而政府部门则能利用智慧旅游平台实现对旅游资源的科学规划和精细管理,提高监管效率和质量。例如,通过实时监控和数据分析,政府可以迅速应对旅游高峰期的客流压力,有效预防景区超载,保障游客安全。同时,智慧旅游还促进了跨行业、跨部门的数据共享与协同合作,为旅游业的可持续发展奠定了坚实基础。总之,智慧旅游以其独特的魅力和无限潜力,正引领着旅游业迈向一个更加智慧、便捷、高效的新时代。

    工业自动化中模拟量滤波防抖PLC程序的实现与应用

    内容概要:本文详细介绍了如何通过PLC程序实现模拟量滤波防抖,确保电流、电压和热电阻等信号的准确采集。核心算法采用掐头去尾平均法,即去掉一组数据中的最大值和最小值后取剩余数据的平均值,以消除因环境干扰导致的异常值。文中提供了详细的代码实现步骤,包括数据结构定义、主程序逻辑、间接寻址方法以及参数配置。此外,还讨论了如何通过死区判断和上升率限制进一步优化滤波效果,提高系统的稳定性和响应速度。 适合人群:从事工业自动化领域的工程师和技术人员,尤其是熟悉PLC编程和模拟量信号处理的专业人士。 使用场景及目标:适用于需要高精度模拟量信号采集的工业控制系统,如电力、化工、制造业等领域。主要目标是提升数据采集的准确性和稳定性,减少外部干扰带来的误差。 其他说明:文中提供的代码示例基于西门子S7-1200/1500系列PLC,但相关原理和方法同样适用于其他品牌的PLC。建议在实际应用中根据具体情况调整参数设置,以达到最佳效果。

    【人工智能大模型发展】从技术突破到场景落地:大模型发展图谱与DeepSeek创新应用解析

    内容概要:本文详细介绍了大模型的发展现状与未来趋势,尤其聚焦于DeepSeek这一创新应用。文章首先回顾了人工智能的定义、分类及其发展历程,指出从摩尔定律到知识密度提升的转变,强调了大模型知识密度的重要性。随后,文章深入探讨了DeepSeek的发展路径及其核心价值,包括其推理模型、思维链技术的应用及局限性。此外,文章展示了DeepSeek在多个行业的应用场景,如智能客服、医疗、金融等,并分析了DeepSeek如何赋能个人发展,具体体现在公文写作、文档处理、知识搜索、论文写作等方面。最后,文章展望了大模型的发展趋势,如通用大模型与垂域大模型的协同发展,以及本地部署小模型成为主流应用渠道的趋势。 适合人群:对人工智能和大模型技术感兴趣的从业者、研究人员及希望利用DeepSeek提升工作效率的个人用户。 使用场景及目标:①了解大模型技术的最新进展和发展趋势;②掌握DeepSeek在不同领域的具体应用场景和操作方法;③学习如何通过DeepSeek提升个人在公文写作、文档处理、知识搜索、论文写作等方面的工作效率;④探索大模型在特定行业的应用潜力,如医疗、金融等领域。 其他说明:本文不仅提供了理论知识,还结合实际案例,详细介绍了DeepSeek在各个场景下的应用方式,帮助读者更好地理解和应用大模型技术。同时,文章也指出了当前大模型技术面临的挑战,如模型的局限性和数据安全问题,鼓励读者关注技术的持续改进和发展。

    电力负荷预测中LSSVM及其改进算法的性能对比研究

    内容概要:本文详细比较了四种基于最小二乘支持向量机(LSSVM)的短期电力负荷预测算法:原始LSSVM、SSA-LSSVM、VMD-LSSVM以及VMD-SSA-LSSVM。通过对这些算法的具体实现和性能评估,展示了每种方法的优势和局限性。实验结果显示,随着算法复杂度的增加,预测精度显著提高,特别是VMD-SSA-LSSVM在RMSE和MAPE等评价指标上表现出色,达到了接近真实值的预测效果。然而,这也伴随着计算成本的大幅上升。 适合人群:从事电力系统调度、数据分析、机器学习领域的研究人员和技术人员。 使用场景及目标:适用于需要进行短期电力负荷预测的研究项目或实际应用,旨在提高预测准确性,减少因天气变化、节假日等因素带来的不确定性影响。 其他说明:文中提供了详细的Python代码片段,帮助读者理解和复现相关算法。同时提醒,在选择模型时需综合考虑预测精度与计算效率之间的平衡。

    基于Python+Django的电影推荐系统:融合机器学习与深度学习的全栈实现

    内容概要:本文详细介绍了一种基于Python和Django框架构建的电影推荐系统。该系统不仅涵盖了用户端的基本功能(如登录、搜索、浏览、评论、评分、收藏),还包括管理端的增删改查操作。后端使用Python和Django框架,结合MySQL数据库,前端采用HTML、CSS和JavaScript实现交互界面。推荐算法方面,利用机器学习和深度学习技术,特别是协同过滤和内容过滤相结合的方式,确保推荐结果的多样性和精准性。此外,文中还讨论了一些常见的技术挑战及其解决方案,如用户冷启动问题、前端交互效果优化、数据库配置错误等。 适合人群:具有一定编程经验的Web开发者和技术爱好者,尤其是对Django框架、机器学习和深度学习感兴趣的读者。 使用场景及目标:适用于希望深入了解并实现一个完整的电影推荐系统的个人或团队。主要目标是掌握如何整合前后端技术,运用机器学习和深度学习算法提升用户体验。 其他说明:文中提供了大量代码片段和实践经验,帮助读者更好地理解和实施各个技术细节。同时强调了系统优化的重要性,如通过Redis缓存提高查询效率,使用AJAX实现无缝加载等。

    MATLAB实现V2G光储充一体化微网多目标优化调度策略及其应用

    内容概要:本文探讨了基于MATLAB平台的V2G(车辆到电网)光储充一体化微网多目标优化调度策略。该策略旨在通过建立光伏微网中以经济性和并网负荷波动率为双目标的蓄电池和V2G协同调度模型,利用粒子群优化(PSO)算法求解模型。文中详细介绍了模型搭建、核心算法实现、运行模式对比以及算例分析。结果显示,V2G模式能够显著提高系统的经济性和稳定性,减少蓄电池的需求量,优化三方(电网、微网调度中心、电动汽车用户)的利益。 适合人群:从事电力系统优化、智能电网研究的专业人士,尤其是对MATLAB编程有一定基础的研究人员和技术人员。 使用场景及目标:适用于需要优化光储充一体化微网调度策略的研究机构和企业。目标是在保证系统经济运行的同时,稳定并网负荷,减少波动,从而提升整体性能。 其他说明:代码注释详尽,包含并行计算框架、电池寿命模型和可视化模块等多个亮点。通过实际案例验证,证明了V2G模式的有效性。

    三菱FX3U五轴钻孔机PLC与威纶通触摸屏程序解析及优化技巧

    内容概要:本文详细介绍了三菱FX3U五轴钻孔机的PLC程序和威纶通触摸屏配置,涵盖梯形图编程、IO分配表、参数设置、自动补偿机制以及异常处理等方面。文章通过具体的代码实例展示了如何实现加工循环、参数动态调整、安全防护等功能,并分享了调试过程中遇到的问题及解决方案。此外,还提供了完整的工程文件,便于读者快速理解和应用。 适合人群:从事工业自动化领域的工程师和技术人员,尤其是对三菱PLC和威纶通触摸屏有一定了解的人群。 使用场景及目标:帮助读者掌握五轴钻孔机的控制系统设计方法,提高编程效率和设备稳定性,适用于类似机床控制系统的开发和维护。 其他说明:文中提到的许多技巧和注意事项来源于作者的实际工作经验,对于初学者来说非常有价值。同时,提供的完整工程文件可以作为参考模板,节省开发时间和成本。

    matlab开发相关资源.m

    matlab开发相关资源

    a383d-main.zip

    a383d-main.zip

    智慧小区解决方案.pptx

    智慧小区解决方案.pptx

    Seafile 基于 Qt 的 GUI 客户端

    Seafile 基于 Qt 的 GUI 客户端

    无人驾驶车辆局部路径规划:基于Matlab的Astar与RRT算法仿真及优化

    内容概要:本文详细介绍了无人驾驶车辆在局部路径规划中的两种经典算法——Astar和RRT的Matlab实现及其优化。首先,文章解释了Astar算法的核心思想,即通过启发函数进行路径搜索,并针对U型障碍等问题提出了双向搜索策略和动态权重调节。接着,文章探讨了RRT算法的特点,如随机生长特性和路径平滑处理,解决了路径过于曲折的问题。此外,还提出了一种混合算法HRA*,通过改进OPEN集的维护方式,提高了算法效率。最后,通过对不同场景的仿真测试,展示了两种算法在复杂环境中的性能差异,并提供了详细的调参经验和优化建议。 适合人群:对无人驾驶技术和路径规划感兴趣的科研人员、工程师以及有一定编程基础的学习者。 使用场景及目标:适用于研究无人驾驶车辆在复杂环境中的路径规划问题,帮助研究人员理解和优化Astar和RRT算法,提高路径规划的效率和准确性。 其他说明:文中附有大量Matlab代码片段和仿真结果图表,便于读者理解和复现实验。同时,提供了关于栅格地图分辨率、车辆动力学参数等方面的实用建议,有助于实际系统的部署和优化。

    选择.txt

    选择

    西门子200Smart与维纶触摸屏在疫苗车间控制系统的应用:配液、发酵、纯化及CIP清洗工艺详解

    内容概要:本文详细介绍了西门子200Smart PLC与维纶触摸屏在某疫苗车间控制系统的具体应用,涵盖配液、发酵、纯化及CIP清洗四个主要工艺环节。文中不仅展示了具体的编程代码和技术细节,还分享了许多实战经验和调试技巧。例如,在配液罐中,通过模拟量处理确保温度和液位的精确控制;发酵罐部分,着重讨论了PID参数整定和USS通讯控制变频器的方法;纯化过程中,强调了双PID串级控制的应用;CIP清洗环节,则涉及复杂的定时器逻辑和阀门联锁机制。此外,文章还提到了一些常见的陷阱及其解决方案,如通讯干扰、状态机切换等问题。 适合人群:具有一定PLC编程基础的技术人员,尤其是从事工业自动化领域的工程师。 使用场景及目标:适用于需要深入了解PLC与触摸屏集成控制系统的工程师,帮助他们在实际项目中更好地理解和应用相关技术和方法,提高系统的稳定性和可靠性。 其他说明:文章提供了大量实战经验和代码片段,有助于读者快速掌握关键技术点,并避免常见错误。同时,文中提到的一些优化措施和调试技巧对提升系统性能非常有帮助。

Global site tag (gtag.js) - Google Analytics