
我是肥哥,一名不专业的面试官!
我是囧囧,一名积极找工作的小菜鸟!
囧囧表示:小白面试最怕的就是面试官问的知识点太笼统,自己无法快速定位到关键问题点!!!
本期主要面试考点
面试官考点之如何用设计模式替换业务场景中复杂的ifelse?


VIP类型
import java.util.Objects;
/**
* @author: 欢迎关注喂信公猪号:囧么肥事
* @date: 2021/12/16
* @email: jiongmefeishi@163.com
*
* 会员类型
*/
public enum VIPEnums {
GOLD(1, "黄金会员"),
STAR(2, "星钻会员"),
SPORTS(3, "体育会员"),
FUN_VIP(4, "FUN会员");
private final int code;
private final String desc;
VIPEnums(int code, String desc) {
this.code = code;
this.desc = desc;
}
public int getCode() {
return code;
}
public String getDesc() {
return desc;
}
public static VIPEnums getByCode(Integer code) {
for (VIPEnums s : VIPEnums.values()) {
if (Objects.equals(s.getCode(), code)) {
return s;
}
}
return null;
}
}
VIP实体
/**
* @author: 欢迎关注喂信公猪号:囧么肥事
* @date: 2021/12/16
* @email: jiongmeifeishi@163.com
*
* vip
*/
public class VIP {
private VIPEnums vipType;
// TODO VIP 其他属性 id, name ...
public VIP() {
}
public VIP(VIPEnums vipType) {
this.vipType = vipType;
}
public VIPEnums getVipType() {
return vipType;
}
public void setVipType(VIPEnums vipType) {
this.vipType = vipType;
}
}
if-else 模式
// if-else 模式
public class App {
public static void main( String[] args ) {
// 黄金会员
VIP vip = new VIP(VIPEnums.GOLD);
if (vip.getVipType().getCode() == VIPEnums.GOLD.getCode()) {
// TODO 黄金会员权益
} else if (vip.getVipType().getCode() == VIPEnums.STAR.getCode()) {
// TODO 星钻会员权益
} else if (vip.getVipType().getCode() == VIPEnums.SPORTS.getCode()) {
// TODO 体育会员权益
} else if (vip.getVipType().getCode() == VIPEnums.FUN_VIP.getCode()) {
// TODO FUN会员权益
} else {
// TODO 其他会员...
}
}
}
VIP策略接口
/**
* @author: 欢迎关注喂信公猪号:囧么肥事
* @date: 2021/12/16
* @email: jiongmefeishi@163.com
* VIP 策略接口
*/
public interface VIPStrategy {
// VIP 具备的权益
void equity();
}
策略接口具体实现类-黄金会员
/**
* @author: 欢迎关注喂信公猪号:囧么肥事
* @date: 2021/12/16
* @email: jiongmefeishi@163.com
*
* 策略接口具体实现类-黄金会员
*/
public class GoldVIPStrategyImpl implements VIPStrategy {
@Override
public void equity() {
// TODO 黄金会员具备的具体权益
}
}
策略接口具体实现类-星钻会员
/**
* @author: 欢迎关注喂信公猪号:囧么肥事
* @date: 2021/12/16
* @email: jiongmefeishi@163.com
*
* 策略接口具体实现类-星钻会员
*/
public class StarVIPStrategyImpl implements VIPStrategy {
@Override
public void equity() {
// TODO 星钻会员具备的具体权益
}
}
策略接口具体实现类-体育会员
/**
* @author: 欢迎关注喂信公猪号:囧么肥事
* @date: 2021/12/16
* @email: jiongmefeishi@163.com
*
* 策略接口具体实现类-体育会员
*/
public class SportsVIPStrategyImpl implements VIPStrategy {
@Override
public void equity() {
// TODO 体育会员具备的具体权益
}
}
策略接口具体实现类-FUN会员
/**
* @author: 欢迎关注喂信公猪号:囧么肥事
* @date: 2021/12/16
* @email: jiongmefeishi@163.com
*
* 策略接口具体实现类-FUN会员
*/
public class FunVIPStrategyImpl implements VIPStrategy {
@Override
public void equity() {
// TODO FUN会员具备的具体权益
}
}
策略上下文类
/**
* @author: 欢迎关注喂信公猪号:囧么肥事
* @date: 2021/12/16
* @email: jiongmefeishi@163.com
*
* 策略上下文类( vip 策略接口的持有者)
*/
public class VIPStrategyContext {
private VIPStrategy vipStrategy;
// 设置VIP策略
public void setVipStrategy(VIPStrategy vipStrategy) {
this.vipStrategy = vipStrategy;
}
// 执行 VIP 权益
public void handle() {
if (vipStrategy != null) {
vipStrategy.equity();
}
}
}
策略工厂
/**
* @author: 欢迎关注喂信公猪号:囧么肥事
* @date: 2021/12/16
* @email: jiongmefeishi@163.com
*
* VIP策略工厂
*/
public class VIPStrategyFactory {
private VIPStrategyFactory() {
}
public static VIPStrategy getVipStrategy(VIP vip) {
VIPStrategy vipStrategy = null;
if (vip.getVipType().getCode() == VIPEnums.GOLD.getCode()) {
// 黄金会员策略实现类
vipStrategy = new GoldVIPStrategyImpl();
} else if (vip.getVipType().getCode() == VIPEnums.STAR.getCode()) {
// 星钻会员策略实现类
vipStrategy = new StarVIPStrategyImpl();
} else if (vip.getVipType().getCode() == VIPEnums.SPORTS.getCode()) {
// 体育会员策略实现类
vipStrategy = new SportsVIPStrategyImpl();
} else if (vip.getVipType().getCode() == VIPEnums.FUN_VIP.getCode()) {
// FUN会员策略实现类
vipStrategy = new FunVIPStrategyImpl();
} else {
// 其他会员...
}
return vipStrategy;
}
}
模拟会员登录获取权益
/**
* @author: 欢迎关注喂信公猪号:囧么肥事
* @date: 2021/12/16
* @email: jiongmefeishi@163.com
*
* 模拟会员登录获取权益
*/
public class TestStrategy {
public static void main(String[] args) {
// 黄金会员
VIP vip = new VIP(VIPEnums.GOLD);
// 策略上下文,执行者
VIPStrategyContext context = new VIPStrategyContext();
// 根据会员类型,获取会员具体策略,获取黄金会员策略
VIPStrategy strategy = VIPStrategyFactory.getVipStrategy(vip);
// 绑定给执行者
context.setVipStrategy(strategy);
// 执行黄金会员的策略,黄金权益
context.handle();
}
}
我们知道, 策略模式的本身设计出来的目的是封装一系列的算法,这些算法都具有共性,可以相互替换,算法独立于使用它的客户端独立变化,客户端不需要了解关注算法的具体实现,客户端仅仅依赖于策略接口 。
通过使用策略模式和工厂模式结合,是不是感觉变得高大上起来了呢??
当然了,最主要的是程序的扩展来说更方便了一些,更符合开闭原则,开放扩展,关闭修改。无论新增多少种新类型的会员,每个人只需要去继承策略接口,实现新会员应有的权益即可。
注意,虽然利于扩展,但是策略模式的缺点也很明显,策略工厂在创建具体的策略实现类的时候,还是书写大量的 if-else 去进行判断,如图

有小伙伴就说了这和不使用策略模式和工厂模式似乎差不多???
抽出一个方法或者封装成一个对象去调用岂不是更简单???

接下来,我们就说说如何优化策略工厂。
首先,我们的工厂,是根据当前传入的用户的会员类型,判断后,返回相应的策略实现类,那么可以借助集合来存储实现类,会员类型作为 key,将所有的会员策略都注册到 map 中。需要注意的是,日常开发基于Spring进行bean管理,上面需要创建的策略类,当然都是希望被 Spring 动态托管,而不是我们自己去一个个的new 出实例。
问题是,如何去实现策略类通过spring进行托管注册?
Spring种提供的InitializingBean接口,这个接口为Bean提供了属性初始化后的处理方法,它只包括afterPropertiesSet方法,凡是继承该接口的类,在bean的属性初始化后都会执行该方法。我们利用此方法把Spring通过IOC创建出来的Bean注册Map 中。
改造策略工厂
import org.example.model.VIP;
import org.example.strategy.VIPStrategy;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
/**
* @author: 欢迎关注喂信公猪号:囧么肥事
* @date: 2021/12/16
* @email: jiongmefeishi@163.com
* <p>
* VIP策略工厂
*/
public class VIPStrategyFactory {
// 存储策略类实例
public static Map<Integer, VIPStrategy> strategyMap = new ConcurrentHashMap<>();
private VIPStrategyFactory() {
}
public static VIPStrategy getVipStrategy(VIP vip) {
if (vip == null) {
return null;
}
return strategyMap.get(vip.getVipType().getCode());
}
}
改造策略类,在bean属性初始化后,将实例对象注册到工厂类中的 map
以黄金会员为例:
import org.example.factory.VIPStrategyFactory;
import org.example.model.VIPEnums;
import org.example.strategy.VIPStrategy;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.stereotype.Service;
/**
* @author: 欢迎关注喂信公猪号:囧么肥事
* @date: 2021/12/16
* @email: jiongmefeishi@163.com
*
* 策略接口具体实现类-黄金会员
*/
@Service
public class GoldVIPStrategyImpl implements VIPStrategy, InitializingBean {
@Override
public void equity() {
// TODO 黄金会员具备的具体权益
System.out.println("黄金会员具备的具体权益");
}
@Override
public void afterPropertiesSet() throws Exception {
VIPStrategyFactory.strategyMap.put(VIPEnums.GOLD.getCode(), new GoldVIPStrategyImpl());
}
}
通过策略模式、工厂模式以及Spring的InitializingBean接口,算是解决了大量的if else,后续新VIP出现也更容易扩展,当然了,这里只是对于设计模式思想的一个简单的示例,实际应用开发中,还是要根据具体的业务场景灵活变通。有需要的小伙伴也可以自己手动模拟一些场景,比如奶茶店各种奶茶新品等等。如果想用囧囧的示例,可公猪号上回复220110 自行导入示例运行即可。
注意:学习软件设计原则,千万不能形成强迫症。当碰到业务复杂的场景时,需要随机应变。
学习设计原则是学习设计模式的基础。在实际开发过程中,并不是一定要求所有代码都遵循设计原则,而是要综合考虑人力、时间、成本、质量,不刻意追求完美,要在适当的场景遵循设计原则。这体现的是一种平衡取舍,可以帮助我们设计出更加优雅的代码结构。
设计模式其实也是一门艺术。设计模式源于生活,不要为了套用设计模式而使用设计模式。

喜欢的小伙伴,欢迎点赞收藏关注

我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co
在railstutorial中,作者为什么选择使用这个(代码list10.25):http://ruby.railstutorial.org/chapters/updating-showing-and-deleting-usersnamespace:dbdodesc"Filldatabasewithsampledata"task:populate=>:environmentdoRake::Task['db:reset'].invokeUser.create!(:name=>"ExampleUser",:email=>"example@railstutorial.org",:passwo
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
鉴于我有以下迁移:Sequel.migrationdoupdoalter_table:usersdoadd_column:is_admin,:default=>falseend#SequelrunsaDESCRIBEtablestatement,whenthemodelisloaded.#Atthispoint,itdoesnotknowthatusershaveais_adminflag.#Soitfails.@user=User.find(:email=>"admin@fancy-startup.example")@user.is_admin=true@user.save!ende
我将应用程序升级到Rails4,一切正常。我可以登录并转到我的编辑页面。也更新了观点。使用标准View时,用户会更新。但是当我添加例如字段:name时,它不会在表单中更新。使用devise3.1.1和gem'protected_attributes'我需要在设备或数据库上运行某种更新命令吗?我也搜索过这个地方,找到了许多不同的解决方案,但没有一个会更新我的用户字段。我没有添加任何自定义字段。 最佳答案 如果您想允许额外的参数,您可以在ApplicationController中使用beforefilter,因为Rails4将参数
给定一个复杂的对象层次结构,幸运的是它不包含循环引用,我如何实现支持各种格式的序列化?我不是来讨论实际实现的。相反,我正在寻找可能会派上用场的设计模式提示。更准确地说:我正在使用Ruby,我想解析XML和JSON数据以构建复杂的对象层次结构。此外,应该可以将该层次结构序列化为JSON、XML和可能的HTML。我可以为此使用Builder模式吗?在任何提到的情况下,我都有某种结构化数据-无论是在内存中还是文本中-我想用它来构建其他东西。我认为将序列化逻辑与实际业务逻辑分开会很好,这样我以后就可以轻松支持多种XML格式。 最佳答案 我最
目录前言滤波电路科普主要分类实际情况单位的概念常用评价参数函数型滤波器简单分析滤波电路构成低通滤波器RC低通滤波器RL低通滤波器高通滤波器RC高通滤波器RL高通滤波器部分摘自《LC滤波器设计与制作》,侵权删。前言最近需要学习放大电路和滤波电路,但是由于只在之前做音乐频谱分析仪的时候简单了解过一点点运放,所以也是相当从零开始学习了。滤波电路科普主要分类滤波器:主要是从不同频率的成分中提取出特定频率的信号。有源滤波器:由RC元件与运算放大器组成的滤波器。可滤除某一次或多次谐波,最普通易于采用的无源滤波器结构是将电感与电容串联,可对主要次谐波(3、5、7)构成低阻抗旁路。无源滤波器:无源滤波器,又称
项目介绍随着我国经济迅速发展,人们对手机的需求越来越大,各种手机软件也都在被广泛应用,但是对于手机进行数据信息管理,对于手机的各种软件也是备受用户的喜爱小学生兴趣延时班预约小程序的设计与开发被用户普遍使用,为方便用户能够可以随时进行小学生兴趣延时班预约小程序的设计与开发的数据信息管理,特开发了小程序的设计与开发的管理系统。小学生兴趣延时班预约小程序的设计与开发的开发利用现有的成熟技术参考,以源代码为模板,分析功能调整与小学生兴趣延时班预约小程序的设计与开发的实际需求相结合,讨论了小学生兴趣延时班预约小程序的设计与开发的使用。开发环境开发说明:前端使用微信微信小程序开发工具:后端使用ssm:VU
我在加密来self正在使用的第三方供应商的值时遇到问题。他们的指令如下:1)Converttheencryptionpasswordtoabytearray.2)Convertthevaluetobeencryptedtoabytearray.3)Theentirelengthofthearrayisinsertedasthefirstfourbytesontothefrontofthefirstblockoftheresultantbytearraybeforeencryption.4)EncryptthevalueusingAESwith:1.256-bitkeysize,2.25
了解Rails缓存如何工作的人可以真正帮助我。这是嵌套在Rails::Initializer.runblock中的代码:config.after_initializedoSomeClass.const_set'SOME_CONST','SOME_VAL'end现在,如果我运行script/server并发出请求,一切都很好。然而,在我的Rails应用程序的第二个请求中,一切都因单元化常量错误而变得糟糕。在生产模式下,我可以成功发出第二个请求,这意味着常量仍然存在。我已通过将以上内容更改为以下内容来解决问题:config.after_initializedorequire'some_cl