成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Spring Boot 開發中有七件事,你必須知道

開發 前端
Controller控制器僅負責處理HTTP請求和響應。業務邏輯應放置在其他層(如Service層)。將業務邏輯與請求和響應處理混合在一起對編寫單元測試非常不利。如果將業務邏輯移動到服務層,那么單元測試可以更加針對服務層進行。

環境:SpringBoot3.2.5

1. 簡介

這篇文章將逐一探討在SpringBoot開發中容易被忽視的7個關鍵細節,從而避免開發過程中的陷阱。

無論是你是初學者還是有經驗的開發者,關注這些小細節往往能夠預防許多常見問題,同時提高開發效率,減少開發過程中的重復工作,甚至可能提升所開發產品的質量。

2. 核心關鍵點

2.1 字段避免@Autowired注入

@Autowired可以將依賴注入到組件中,但過度使用它可能會導致緊密的耦合和測試困難。使用構造器注入或@Resource等方法可以使依賴關系更加清晰。

推薦做法:

優先使用構造器注入,因為它可以清晰地定義組件的依賴,并且在單元測試中更容易進行模擬(mock)。

如果你當前使用了Lombok,你可以利用@RequiredArgsConstructor注解來自動生成構造器。

private final UserRepository userRepository ;
public UserService(UserRepository userRepository) {
  this.userRepository = userRepository ;
}

我們是禁止使用Lombok的。

最后,我還是推薦使用構造函數注入,避免字段上使用@Autowired / @Resource注解,并且Spring官方推薦的也是構造函數注入。

2.2 避免在控制器中編寫業務邏輯

嚴格來說,Controller控制器僅負責處理HTTP請求和響應。業務邏輯應放置在其他層(如Service層)。將業務邏輯與請求和響應處理混合在一起對編寫單元測試非常不利。如果將業務邏輯移動到服務層,那么單元測試可以更加針對服務層進行。

推薦做法:

將業務邏輯移動到服務層(Service),并讓控制器僅處理請求并調用服務方法。進行這種分離后,不僅單元測試更加方便,代碼也更容易重用。

@RestController
@RequestMapping("/products")
public class ProductController {


  private final ProductService productService ;
  public ProductController(ProductService productService) {
    this.productService = productService ;
  }


  @GetMapping("/{id}")
  public ResponseEntity<Product> getProduct(@PathVariable Long id) {
    // 調用Service進行業務邏輯的處理
    Product product = productService.getProductById(id) ;
    return ResponseEntity.ok(product) ;
  }
}

這也是我們日常開發中最基本的要求了。

2.3 使用@ConfigurationProperties替代@Value

使用@Value注解來獲取配置雖然簡單,但缺乏結構性。此外,過度使用會導致@Value注解散布在整個項目中,這不利于代碼的維護和重用。使用@ConfigurationProperties可以避免這些問題,使配置更清晰、更易于管理。

推薦做法:

創建一個專用的配置類,并使用@ConfigurationProperties注解來綁定相關的配置項,這增強了代碼的可讀性。當在多個地方使用相同的配置類時,它有助于避免重復配置屬性,從而提高了代碼的可重用性。這種方法還使配置更具結構性,便于維護和理解。

例如,在應用配置的情況下,當處理大量屬性或復雜配置結構時,@ConfigurationProperties所提供的便利性和長期影響遠遠超過了創建一個新類所需的工作量。

@ConfigurationProperties(prefix = "pack.app")
public class AppConfig {
  private String title ;
  private String version ;
  private Integer uid ;
  // getters and setters
}

2.4 避免構造函數過于復雜

構造器應盡可能保持簡單。做過多的初始化工作會使構造器變得復雜且難以理解。此外,如果構造器中做了太多工作,未來的需求變更很可能需要頻繁修改,從而增加了代碼維護的難度。它還顯著影響性能,因為在對象創建期間執行了復雜操作。

推薦做法:

主要使用構造器進行依賴注入,并將初始化工作移動到用@PostConstruct注解的方法中或在服務方法內執行。如果必須在構造器中執行大量操作,考慮實現延遲加載或將其轉換為工廠模式。

public class CommonComponent {
  private final CommonService commonService ;


  public CommonComponent(CommonService commonService) {
    this.commonService = commonService ;
  }


  @PostConstruct
  public void init() {
    // TODO
  }
}

構造器只做基本的注入操作。其它初始化的工作通過@PostConstruct注解的方法來處理。

2.5 定義不同的環境配置文件

為不同環境(開發、測試、生產)使用不同的配置有助于隔離環境差異。

推薦做法:

使用application-{profile}.properties或application-{profile}.yml來為每個環境定義配置。

激活不同的配置文件:

spring:
  profiles:
    active:
    - dev

2.6 使用異常替代返回值

首先,先來看段代碼:

@Service
public class ProductService {


  private final ProductRepository productRepository ;
  public ProductService(ProductRepository productRepository) {
    this.productRepository = productRepository ;
  }


  public R<Product, String> queryById(Long id) {
    Optional<Product> opt = productRepository.findById(id) ;
    if (opt.isPresent()) {
      return R.success(opt.get()) ;
    } else {
      return R.error("商品不存在id: " + id) ;
    }
  }
}

上面代碼直接使用R作為方法的返回值顯得不夠優雅。

如果將返回R.error的部分替換為拋出new XXException異常,這不僅能提高代碼的可讀性,還能讓服務返回業務結果,而不是與控制器結果糾纏在一起。

優化代碼:

@Service
public class ProductService {


  private final ProductRepository productRepository ;
  public ProductService(ProductRepository productRepository) {
    this.productRepository = productRepository ;
  }


  public Product queryById(Long id) {
    return productRepository.findById(id)
      .orElseThrow(() -> new ProductNotFoundException("商品不存在id: " + id));
  }
}

Service層應該只返回業務結果,而不應涉及控制器的結果。此外,拋出的異常可以讓維護人員立即理解,并指出問題所在。

最后,我們利用@RestControllerAdvice注解來進行全局異常處理,以便及時捕獲業務邏輯中拋出的異常,避免500錯誤,如下示例:

@RestControllerAdvice
public class GlobalExceptionHandler {


  @ExceptionHandler(ProductNotFoundException.class)
  public ResponseEntity<ErrorResponse> handleProductNotFoundException(ProductNotFoundException ex) {
    ErrorResponse errorResponse = new ErrorResponse(HttpStatus.NOT_FOUND.value(), ex.getMessage()) ;
    return new ResponseEntity<>(errorResponse, HttpStatus.NOT_FOUND) ;
  }
  // 其它異常處理句柄
}

有人可能會說這種拋異常處理方式是反模式設計,你覺得呢?

2.7 優先考慮ResponseEntity作為響應

很多人會自定義對象作為Controller接口返回的統一對象,但SpringBoot本身提供了一個專門的響應實體,即ResponseEntity。

ResponseEntity提供了更大的靈活性,允許控制響應的各個方面,包括HTTP狀態碼、響應頭、響應體等。這使得程序能夠更精確地構建響應結果,根據業務需求返回不同的HTTP狀態碼。

此外,ResponseEntity還支持泛型,允許返回不同類型的響應體,滿足各種業務場景下的響應需求。

下面是ResponseEntity API文檔說明:

圖片圖片

這點并非必須遵守,當你確實需要高度定制化,那么使用自定義的結果對象也當然沒有問題。

責任編輯:武曉燕 來源: Spring全家桶實戰案例源碼
相關推薦

2016-12-01 14:54:57

2010-07-28 14:21:43

Flex

2015-05-29 09:45:42

Google IOA

2017-07-04 08:59:35

2017-04-26 16:24:49

路由器5GHz頻段

2015-04-09 09:53:08

CA TechnoloDevOps

2018-08-01 22:14:23

Kubernetes容器云遷移

2016-11-21 11:50:37

2022-02-24 14:22:39

首席信息官CIONFT

2011-12-16 17:05:58

2015-03-30 15:00:34

PHPPHP五件事

2015-03-20 16:12:23

2011-08-18 16:34:28

程序員必須知道

2015-08-11 17:55:21

谷歌重組科技

2012-02-07 13:29:35

2010-04-12 14:58:56

Meego開發

2015-01-22 10:27:55

win10

2015-08-26 14:07:02

創業關于創業

2022-02-28 15:28:43

開源云時代軟件
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 91精品国产综合久久久久 | 国产精品性做久久久久久 | 毛片大全 | 天天爽一爽 | 亚洲精品一区在线观看 | 色婷综合网 | 精品国产乱码一区二区三 | 欧美日韩国产一区二区三区不卡 | 一区视频在线 | 国产三级| 中文字幕一区二区三区四区五区 | 日韩一区二区在线视频 | 日韩国产欧美 | 亚洲精品性视频 | 国产高清视频在线观看 | 日本电影一区二区 | 91精品国产日韩91久久久久久 | 国产精品揄拍一区二区 | 美女久久| 超碰免费在 | 国产精品视频久久久久 | 亚洲精品久久久久久首妖 | 久久精品亚洲欧美日韩久久 | 韩日视频在线观看 | 日日摸日日碰夜夜爽亚洲精品蜜乳 | 天天色天天| 精久久久久 | 日本中文字幕在线观看 | 久色网 | 亚欧洲精品在线视频免费观看 | 日韩中文字幕2019 | 天天操天天射天天 | 欧美精品久久久 | 免费成人国产 | 日本三级黄视频 | 国产一级视频在线播放 | 亚洲国产成人精品一区二区 | 国产欧美精品一区二区三区 | 亚洲视频免费观看 | 福利视频一区二区 | 91精品国产综合久久香蕉麻豆 |