亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Spring如何消除代碼中的if-else/switch-case

 更新時(shí)間:2019年04月09日 09:14:11   作者:水目沾  
這篇文章主要給大家介紹了關(guān)于Spring如何消除代碼中if-else/switch-case的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家學(xué)習(xí)或者使用Spring具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧

前言

在很多時(shí)候,我們代碼中會(huì)有很多分支,而且分支下面的代碼又有一些復(fù)雜的邏輯,相信很多人都喜歡用 if-else/switch-case 去實(shí)現(xiàn)。做的不好的會(huì)直接把實(shí)現(xiàn)的代碼放在 if-else/switch-case 的分支之下:

switch ( type ) {
 case case1:
  ...
  ...
  break;
 case case2:
  ...
  ...
  break;
 case case3:
  ...
  ...
  break
 default:
  return null;
}

這樣的代碼不僅冗長(zhǎng),讀起來(lái)也非常困難。做的好一點(diǎn)的會(huì)把這些邏輯封裝成函數(shù)然后在分支中調(diào)用:

switch ( type ) {
 case case1:
  return case1Func();
 case case2:
  return case2Func();
 case case3:
  return case3Func();
 default:
  return null;
}

即使這樣也是面向過(guò)程思維的寫法,以前寫 C 程序的時(shí)候也總喜歡這樣寫,毫無(wú)設(shè)計(jì)模式可言。不僅違背開(kāi)閉原則,而且隨著 switch-case 分支的增多,該段代碼只會(huì)越來(lái)越冗長(zhǎng)。其實(shí)這種代碼已經(jīng)有成熟的模式去消除諸多的 if-else/switch-case 分支。本文就教大家在 Spring 中如何用注解+策略模式+簡(jiǎn)單工廠的方式消除 if-else/switch-case 。我們就拿 QQ 空間的個(gè)人中心舉例子,假如 QQ 空間個(gè)人中心有四個(gè) tab 分別是列出我的說(shuō)說(shuō)、我的日志、我的照片和我的訪客。一般的后臺(tái)代碼很有可能如下:

//各個(gè) tab 名稱的枚舉:
public enum UserRelatedType {
 /**
  * 說(shuō)說(shuō)
  */
 SHUOSHUO("說(shuō)說(shuō)"),

 /**
  * 日志
  */
 RIZHI("日志"),

 /**
  * 發(fā)布
  */
 ZHAOPIAN("照片"),

 /**
  * 訪客
  */
 FANGKE("");

 private String desc;

 UserRelatedType(String desc) {
  this.desc = desc;
 }

 public String getDesc() {
  return desc;
 }

 public void setDesc(String desc) {
  this.desc = desc;
 }
}

列出 QQ 用戶個(gè)人中心相關(guān) tab 的代碼:

public List<UserRelatedVO> listRelated(UserRelatedQuery query){

 UserRelatedType relatedType = UserRelatedType.valueOf(StringUtils.upperCase(query.getType()) );
 switch ( relatedType ) {
  case SHUOSHUO:
   return listRelatedShuoshuo( query );
  case RIZHI:
   return listRelatedRizhi( query );
  case ZHAOPIAN:
   return listRelatedZhaopian( query );
  case FANGKE:
   return listRelatedFangke( query );
  default:
   return null;
 }
}

而采用注解+策略模式+簡(jiǎn)單工廠,重構(gòu)后代碼如下:

1、定義一個(gè)注解,用來(lái)完全消除 if-else:

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface RelatedTypeAnnotation {
 /**
  * 用戶相關(guān)類型名稱
  */
 UserRelatedType value();
}

2、先定義了個(gè)接口,所有 tab 都要實(shí)現(xiàn)該接口。其中 list 是 tab 數(shù)據(jù)展示的方法。

public interface UserRelated {

 /**
  * 列出詳細(xì)信息
  *
  * @param query
  * @return
  */
 List<UserRelatedVO> list(UserRelatedQuery query);
}

3、定義具體的各個(gè) tab 的實(shí)現(xiàn),繼承 UserRelated 策略接口

我的說(shuō)說(shuō)

@Component("userRelatedShuoshuo")
@RelatedTypeAnnotation( value = UserRelatedType.SHUOSHUO )
public class UserRelatedShuoshuo implements UserRelated {
 @Override
 public List<UserRelatedVO> list(UserRelatedQuery query) {
  System.out.println("我的說(shuō)說(shuō)!");
  return list;
 }
}

我的日志

@Component("userRelatedRizhi")
@RelatedTypeAnnotation( value = UserRelatedType.RIZHI )
public class UserRelatedRizhi implements UserRelated {
 @Override
 public List<UserRelatedVO> list(UserRelatedQuery query) {
  System.out.println("我的日志!");
  return list;
 }
}

我的照片

@Component("userRelatedZhaopian")
@RelatedTypeAnnotation( value = UserRelatedType.ZHAOPIAN )
public class UserRelatedZhaopian implements UserRelated {
 @Override
 public List<UserRelatedVO> list(UserRelatedQuery query) {
  System.out.println("我的照片!");
  return list;
 }
}

我的訪客

@Component("userRelatedFangke")
@RelatedTypeAnnotation( value = UserRelatedType.FANGKE )
public class UserRelatedFangke implements UserRelated {
 @Override
 public List<UserRelatedVO> list(UserRelatedQuery query) {
  System.out.println("我的訪客!");
  return list;
 }
}

3、定義一個(gè)從 Spring context 獲取 bean 的工具類

@Component
public class SpringContextUtil implements ApplicationContextAware {

 private ApplicationContext context;

 public ApplicationContext getContext() {
  return context;
 }

 @Override
 public void setApplicationContext(ApplicationContext context)throws BeansException {
  this.context = context;
 }
}

4、定義一個(gè)簡(jiǎn)單工廠,用來(lái)生產(chǎn)各種 tab 對(duì)象。

@Component
public class UserRelatedFactory {

 @Autowired
 SpringContextUtil springContextUtil;
 
 private static Map<UserRelatedType, UserRelated> userRelatedMap = Maps.newConcurrentMap();

 //工廠將 Spring 裝配的相關(guān)的 Bean 用 Map 保存起來(lái)
 public UserRelatedFactory(){
  Map<String, Object> beanMap = springContextUtil.getContext().getBeansWithAnnotation(RelatedTypeAnnotation.class);

  for(Object userRelated : beanMap.values()) {
   RelatedTypeAnnotation annotation = userRelated.getClass().getAnnotation(RelatedTypeAnnotation.class);
   userRelatedMap.put(annotation.value(), (UserRelated)userRelated);
  }
 }

 public static UserRelated createRelated(UserRelatedType relatedType) {
  return userRelatedMap.get( relatedType );
 }
}

5、調(diào)用的代碼(listRelated 會(huì)在 controller 中被調(diào)用)。

public List<UserRelatedVO> listRelated(UserRelatedQuery query){

 UserRelatedType relatedType = UserRelatedType.valueOf(StringUtils.upperCase(query.getType()) );
 UserRelated related = UserRelatedFactory.createRelated( relatedType );
 if( related != null ) {
  return related.list( query );
 } else {
  return null;
 }
}

重構(gòu)后的代碼如果需要再新增一種 tab,比如我的好友,只需要新增一種類型繼承 UserRelated 實(shí)現(xiàn)其中的 list,并加上相應(yīng)的注解即可。

其實(shí)這是一種通用的解決方案,當(dāng)你 if-else/switch-case 的分支超過(guò) 3 個(gè)、且分支代碼相似且冗長(zhǎng)的情況下就應(yīng)該考慮這種模式。這種模式寫出的代碼面向?qū)ο?、清晰、易擴(kuò)展還高大上,何樂(lè)而不為呀,趕緊試試吧!

總結(jié)

以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。

相關(guān)文章

最新評(píng)論