简单聊聊策略模式(新手推荐) 策略模式结构图
yuyutoo 2024-10-20 13:08 1 浏览 0 评论
设计模式是个神奇的东西,用得好的话可以让你的代码结构清晰简洁,用得不好则会让结构更加复杂难懂,包括使用过度。下面我打算分好几部分介绍下自己所理解的设计模式。
最近在看一本关于设计模式的书《深入浅出设计模式》,觉得书中讲的很生动形象。我打算用自己的理解结合书中的例子,给大家简单解释一下,适合和我一样的新手,同时欢迎大神批评指正。首先从策略模式开始。下面大家来看一个例子:
现在有抽象父类People,里面有eat()和sleep()等共有的方法以及抽象方法show()。下面有三个子类man,woman以及child,在子类中我们实现对应的show()方法。似乎并没有什么问题,大家都很好。具体代码如下:
People
public abstract class People {
public void eat(){
System.out.println("all people eat");
}
public void sleep(){
System.out.println("all people sleep");
}
public abstract void show();
}
Man
public class Man extends People{
@Override
public void show() {
System.out.println("I have short hair");
}
}
Woman
public class Woman extends People {
@Override
public void show() {
System.out.println("I have long hair");
}
}
Child
public class Child extends People {
@Override
public void show() {
System.out.println("I have less hair");
}
}
分析以上代码,我们将公有部分(eat和sleep)在抽象类People实现,show在具体子类中实现。(不要说我吃米饭,婴儿吃奶嘴,eat应该不同的,大家安静安静别吵,现在不是纠结这个的时候)只能说我这个例子举的不太恰当,因为实际需求中总有一部分是固定不变的,我们这一步的任务就是将固定不变的抽取出来父类中实现而已。
这个地方有个很重要的思想,就是面对一个系统或者说一个模块,我们将固定不变的部分与经常变化的部分区分开来,分别对待,小心产品经理。
假设我现在需求要变,要所有的人都有move()方法。你会怎么做?你肯定会说我们都是面向对象编程,很简单啊,直接在People中增加move()方法并实现,这种上课时老师一直讲的东西,继承。
可是你想过没有,这样做会不会有什么问题?
child婴儿是不会移动的(不要纠结说婴儿会爬之类的,我眼睛会干)。这时候你会发现如果我们在父类中实现,那么将影响整个子类,这对于子类较多的类来说,是灾难性的。你肯定就会说,我们在People中设计一个抽象方法move(),然后在子类中实现不可以吗?
当然可以,代码如下:
People
public abstract class People {
public void eat(){
System.out.println("all people eat");
}
public void sleep(){
System.out.println("all people sleep");
}
public abstract void show();
public abstract void move();
}
Man
public class Man extends People{
@Override
public void show() {
System.out.println("I have short hair");
}
@Override
public void move() {
System.out.println("on foot");
}
}
Woman
public class Woman extends People {
@Override
public void show() {
System.out.println("I have long hair");
}
@Override
public void move() {
System.out.println("on foot");
}
}
Child
public class Child extends People {
@Override
public void show() {
System.out.println("I have less hair");
}
@Override
public void move() {
}
}
好了,实现完成。在child类中,move()中我们什么也不做。现在需求来了,我们有一类人懒人。虽然是个大老爷们,但是我不想动,我被床封印了。你会怎么做?重新创建一个LazyMan类继承People,然后实现move(),在move中什么也不做?
你会说怎么不可以???
child和LazyMan中代码代码重复,做到代码重用了吗?什么,实现为空也叫代码重复?
住口。在Man和Woman中move不是已经代码重复了吗?还敢嘴硬。少年还有什么可说的??
你会说,看来不行啊,我们必须求助于其他办法了。对的,我们试试其他办法。
能不能用接口?然后创建多个接口的子类分别实现?
具体实现如下:
IMoveBehavior
public interface IMoveBehavior {
public void move();
}
MoveWithFoot
public class MoveWithFoot implements IMoveBehavior {
@Override
public void move() {
System.out.println("on foot");
}
}
MoveNothing
public class MoveNothing implements IMoveBehavior{
@Override
public void move() {
System.out.println("move? no");
}
}
来看看我们的People
public abstract class People {
IMoveBehavior moveBehavior;
public void eat(){
System.out.println("all people eat");
}
public void sleep(){
System.out.println("all people sleep");
}
public abstract void show();
public void moveBehave(){
moveBehavior.move();
}
}
我们在创建IMoveBehavior对象的时候需要知道IMoveBehavior具体指向的是什么,所以我们在People子类的构造函数中来指定。代码如下:
Man
public class Man extends People implements IMoveBehavior{
public Man(IMoveBehavior moveBehavior){
this.moveBehavior = moveBehavior;
}
@Override
public void show() {
System.out.println("I have short hair");
}
@Override
public void move() {
System.out.println("on foot");
}
}
Woman和Child类似,不贴了。我们写个测试类StrategyTest,看我们的代码复用是否成功?
StrategyTest
public class StrategyTest {
public static void main(String[] args) {
Man niceMan = new Man(new MoveNothing());
niceMan.moveBehave();
}
}
运行结果如下:
如果是勤劳的人,那就 Man niceMan = new Man(new MoveWithFoot());
问题解决,那还有没有什么不妥的地方呢?
我们moveBehavior是在创建的时候进行绑定的,如果我想动态地更改moveBehavior呢?是不是就傻眼了?
于是我们有个接下来的方法:
People
public abstract class People {
IMoveBehavior moveBehavior;
public void setMoveBehavior(IMoveBehavior moveBehavior){
this.moveBehavior = moveBehavior;
}
public void eat(){
System.out.println("all people eat");
}
public void sleep(){
System.out.println("all people sleep");
}
public abstract void show();
public void moveBehave(){
moveBehavior.move();
}
}
其他类不变。我们将moveBehavior的设置方法暴露出去,可以进行动态设定。来验证一下:
public class StrategyTest {
public static void main(String[] args) {
Man niceMan = new Man();
//给我走
niceMan.setMoveBehavior(new MoveWithFoot());
niceMan.moveBehave();
//不许动
niceMan.setMoveBehavior(new MoveNothing());
niceMan.moveBehave();
}
}
验证结果如下:
策略模式:
定义了算法族,分别封装起来,让它们之间可以互相替换, 此模式让算法的变化独立于使用算法的客户。
最后说一个重要的点:我们之前那种做法都是通过继承,面向实现做的,虽然可以得到要求,但是代码复用性不高,如果子类多,那将是灾难。我们要面向接口编程。
原则:少用继承,多用组合。
拓展:
1. 假设需求又变了,需要给所有人加上wear(穿衣的方法),但是每种人的穿衣风格不同。男的穿西装,女的穿裙子,小孩穿肚兜,甚至超人内裤外穿。好解决吗?
2. 需求又变了,要求move增加骑自行实现。创造出喜欢骑自行车穿裙子的女生,好解决吗?嘿嘿嘿
最后有必要说一下,我文中出现猥琐的语句只是为了提起大家的兴趣,给大家将知识留下深刻的印象,我本人很正直,不信看我眼睛说话。
大家下期见,拜拜!
相关推荐
- jQuery VS AngularJS 你更钟爱哪个?
-
在这一次的Web开发教程中,我会尽力解答有关于jQuery和AngularJS的两个非常常见的问题,即jQuery和AngularJS之间的区别是什么?也就是说jQueryVSAngularJS?...
- Jquery实时校验,指定长度的「负小数」,小数位未满末尾补0
-
在可以输入【负小数】的输入框获取到焦点时,移除千位分隔符,在输入数据时,实时校验输入内容是否正确,失去焦点后,添加千位分隔符格式化数字。同时小数位未满时末尾补0。HTML代码...
- 如何在pbootCMS前台调用自定义表单?pbootCMS自定义调用代码示例
-
要在pbootCMS前台调用自定义表单,您需要在后台创建表单并为其添加字段,然后在前台模板文件中添加相关代码,如提交按钮和表单验证代码。您还可以自定义表单数据的存储位置、添加文件上传字段、日期选择器、...
- 编程技巧:Jquery实时验证,指定长度的「负小数」
-
为了保障【负小数】的正确性,做成了通过Jquery,在用户端,实时验证指定长度的【负小数】的方法。HTML代码<inputtype="text"class="forc...
- 一篇文章带你用jquery mobile设计颜色拾取器
-
【一、项目背景】现实生活中,我们经常会遇到配色的问题,这个时候去百度一下RGB表。而RGB表只提供相对于的颜色的RGB值而没有可以验证的模块。我们可以通过jquerymobile去设计颜色的拾取器...
- 编程技巧:Jquery实时验证,指定长度的「正小数」
-
为了保障【正小数】的正确性,做成了通过Jquery,在用户端,实时验证指定长度的【正小数】的方法。HTML做成方法<inputtype="text"class="fo...
- jquery.validate检查数组全部验证
-
问题:html中有多个name[],每个参数都要进行验证是否为空,这个时候直接用required:true话,不能全部验证,只要这个数组中有一个有值就可以通过的。解决方法使用addmethod...
- Vue进阶(幺叁肆):npm查看包版本信息
-
第一种方式npmviewjqueryversions这种方式可以查看npm服务器上所有的...
- layui中使用lay-verify进行条件校验
-
一、layui的校验很简单,主要有以下步骤:1.在form表单内加上class="layui-form"2.在提交按钮上加上lay-submit3.在想要校验的标签,加上lay-...
- jQuery是什么?如何使用? jquery是什么功能组件
-
jQuery于2006年1月由JohnResig在BarCampNYC首次发布。它目前由TimmyWilson领导,并由一组开发人员维护。jQuery是一个JavaScript库,它简化了客户...
- django框架的表单form的理解和用法-9
-
表单呈现...
- jquery对上传文件的检测判断 jquery实现文件上传
-
总体思路:在前端使用jquery对上传文件做部分初步的判断,验证通过的文件利用ajaxFileUpload上传到服务器端,并将文件的存储路径保存到数据库。<asp:FileUploadI...
- Nodejs之MEAN栈开发(四)-- form验证及图片上传
-
这一节增加推荐图书的提交和删除功能,来学习node的form提交以及node的图片上传功能。开始之前需要源码同学可以先在git上fork:https://github.com/stoneniqiu/R...
- 大数据开发基础之JAVA jquery 大数据java实战
-
上一篇我们讲解了JAVAscript的基础知识、特点及基本语法以及组成及基本用途,本期就给大家带来了JAVAweb的第二个知识点jquery,大数据开发基础之JAVAjquery,这是本篇文章的主要...
- 推荐四个开源的jQuery可视化表单设计器
-
jquery开源在线表单拖拉设计器formBuilder(推荐)jQueryformBuilder是一个开源的WEB在线html表单设计器,开发人员可以通过拖拉实现一个可视化的表单。支持表单常用控件...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- mybatis plus (70)
- scheduledtask (71)
- css滚动条 (60)
- java学生成绩管理系统 (59)
- 结构体数组 (69)
- databasemetadata (64)
- javastatic (68)
- jsp实用教程 (53)
- fontawesome (57)
- widget开发 (57)
- vb net教程 (62)
- hibernate 教程 (63)
- case语句 (57)
- svn连接 (74)
- directoryindex (69)
- session timeout (58)
- textbox换行 (67)
- extension_dir (64)
- linearlayout (58)
- vba高级教程 (75)
- iframe用法 (58)
- sqlparameter (59)
- trim函数 (59)
- flex布局 (63)
- contextloaderlistener (56)