自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Spring Boot 開發(fā)中有七件事,你必須知道

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

環(huán)境:SpringBoot3.2.5

1. 簡介

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

無論是你是初學(xué)者還是有經(jīng)驗的開發(fā)者,關(guān)注這些小細節(jié)往往能夠預(yù)防許多常見問題,同時提高開發(fā)效率,減少開發(fā)過程中的重復(fù)工作,甚至可能提升所開發(fā)產(chǎn)品的質(zhì)量。

2. 核心關(guān)鍵點

2.1 字段避免@Autowired注入

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

推薦做法:

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

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

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

我們是禁止使用Lombok的。

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

2.2 避免在控制器中編寫業(yè)務(wù)邏輯

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

推薦做法:

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

@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) {
    // 調(diào)用Service進行業(yè)務(wù)邏輯的處理
    Product product = productService.getProductById(id) ;
    return ResponseEntity.ok(product) ;
  }
}

這也是我們?nèi)粘i_發(fā)中最基本的要求了。

2.3 使用@ConfigurationProperties替代@Value

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

推薦做法:

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

例如,在應(yīng)用配置的情況下,當處理大量屬性或復(fù)雜配置結(jié)構(gòu)時,@ConfigurationProperties所提供的便利性和長期影響遠遠超過了創(chuàng)建一個新類所需的工作量。

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

2.4 避免構(gòu)造函數(shù)過于復(fù)雜

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

推薦做法:

主要使用構(gòu)造器進行依賴注入,并將初始化工作移動到用@PostConstruct注解的方法中或在服務(wù)方法內(nèi)執(zhí)行。如果必須在構(gòu)造器中執(zhí)行大量操作,考慮實現(xiàn)延遲加載或?qū)⑵滢D(zhuǎn)換為工廠模式。

public class CommonComponent {
  private final CommonService commonService ;


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


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

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

2.5 定義不同的環(huán)境配置文件

為不同環(huán)境(開發(fā)、測試、生產(chǎn))使用不同的配置有助于隔離環(huán)境差異。

推薦做法:

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

激活不同的配置文件:

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作為方法的返回值顯得不夠優(yōu)雅。

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

優(yōu)化代碼:

@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層應(yīng)該只返回業(yè)務(wù)結(jié)果,而不應(yīng)涉及控制器的結(jié)果。此外,拋出的異??梢宰尵S護人員立即理解,并指出問題所在。

最后,我們利用@RestControllerAdvice注解來進行全局異常處理,以便及時捕獲業(yè)務(wù)邏輯中拋出的異常,避免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) ;
  }
  // 其它異常處理句柄
}

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

2.7 優(yōu)先考慮ResponseEntity作為響應(yīng)

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

ResponseEntity提供了更大的靈活性,允許控制響應(yīng)的各個方面,包括HTTP狀態(tài)碼、響應(yīng)頭、響應(yīng)體等。這使得程序能夠更精確地構(gòu)建響應(yīng)結(jié)果,根據(jù)業(yè)務(wù)需求返回不同的HTTP狀態(tài)碼。

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

下面是ResponseEntity API文檔說明:

圖片圖片

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

責任編輯:武曉燕 來源: Spring全家桶實戰(zhàn)案例源碼
相關(guān)推薦

2016-12-01 14:54:57

2010-07-28 14:21:43

Flex

2015-05-29 09:45:42

Google IOA

2017-04-26 16:24:49

路由器5GHz頻段

2017-07-04 08:59:35

2015-04-09 09:53:08

CA TechnoloDevOps

2016-11-21 11:50:37

2018-08-01 22:14:23

Kubernetes容器云遷移

2022-02-24 14:22:39

首席信息官CIONFT

2011-12-16 17:05:58

2011-08-18 16:34:28

程序員必須知道

2015-03-30 15:00:34

PHPPHP五件事

2015-03-20 16:12:23

2012-02-07 13:29:35

2015-08-11 17:55:21

谷歌重組科技

2010-04-12 14:58:56

Meego開發(fā)

2015-01-22 10:27:55

win10

2022-02-28 15:28:43

開源云時代軟件

2015-08-26 14:07:02

創(chuàng)業(yè)關(guān)于創(chuàng)業(yè)
點贊
收藏

51CTO技術(shù)棧公眾號