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

如何編寫優雅的 Controller 代碼?

開發 前端
看過很多代碼,業務邏輯全部寫在 Controller層,并不能說這樣的做法是錯的,但是看起來很別扭,不優雅!那么,Controller里面應該如何編寫才算優雅呢?

作為一名 Java程序員,對 Controller肯定不陌生,它是與外部客戶端通信的入口,比如常見的 REST 操作(GET、PUT、POST、DELETE等),那么,Controller里面應該如何編寫才算優雅呢?

其實,一個優雅的 Controller,里面的代碼主要包含下面 6個部分:

  • 接收 HTTP(s)請求
  • 解析請求參數
  • 驗證請求參數
  • 調用業務方法
  • 組織返回數據
  • 統一異常處理

下面一一講解這 6個部分:

一、接收 HTTP(s)請求

接收 HTTP(s)請求是 Controller的入口,這里以查詢用戶信息為例進行說明,如下代碼:

@RestController
public class UserController {
    @GetMapping("/user/{userId}")
    public void getUserById(@PathVariable String userId) {
        // 業務邏輯
    }
}

在上面的示例中,我們使用 URL/user/{id}接收用戶發出的 GET請求,然后通過getUserById方法進行真實的業務處理。通過上面的代碼,一個請求就被 Controller層成功接收了。

二、解析請求參數

接收到請求后,一般需要對請求參數進行解析,如下示例代碼:

@RestController
public class UserController {
    @PostMapping("/user/register")
    public void getGradeById(@RequestBody User user) {
        // 代碼邏輯
    }
}

public class User {
    private String nickname;
    private Integer age;
    // getters and setters and constructors
}

上述示例代碼將請求的 body映射到 User對象上,因此,請求的 body體應該是:

{
  "nickname": "huahua",
  "age": "18"
}

在 SpringMVC 中,常見的參數類型及其用途如下:

1.原始 HTTP請求和響應對象

直接接收原始的 HTTP請求和響應對象,HttpServletRequest 和 HttpServletResponse

@RequestMapping("/test")
public void example(HttpServletRequest request, HttpServletResponse response) {
    // 處理請求和響應
}

2.路徑變量 (@PathVariable)

用于獲取 URL 路徑中的動態部分。

@RequestMapping("/user/{id}")
public String getUser(@PathVariable("id") String userId) {
    // 使用 userId 進行處理
    return "userDetail";
}

3.請求參數 (@RequestParam)

用于獲取 URL 查詢參數或表單數據。

@RequestMapping("/search")
public String search(@RequestParam("query") String query) {
    // 使用 query 進行搜索
    return "searchResults";
}

4.請求體 (@RequestBody)

用于接收請求體中的數據,常用于處理 JSON 或 XML 格式的數據。

@RequestMapping(value = "/create", method = RequestMethod.POST)
public String create(@RequestBody User user) {
    // 處理 user 對象
    return "user";
}

5.模型屬性 (@ModelAttribute)

用于綁定表單數據到模型對象。

@RequestMapping("/register")
public String register(@ModelAttribute User user) {
    // 處理 user 對象
    return "user";
}

6.會話屬性 (@SessionAttribute)

用于訪問會話中的屬性。

@RequestMapping("/profile")
public String profile(@SessionAttribute("user") User user) {
    // 處理會話中的 user 對象
    return "profile";
}

7.請求頭 (@RequestHeader)

用于訪問 HTTP 請求頭信息。

@RequestMapping("/headers")
public String headers(@RequestHeader("User-Agent") String userAgent) {
    // 使用 userAgent 進行處理
    return "headerInfo";
}

8.Cookie 值 (@CookieValue)

用于訪問 Cookie 的值。

@RequestMapping("/cookies")
public String cookies(@CookieValue("sessionId") String sessionId) {
    // 使用 sessionId 進行處理
    return sessionId;
}

9.自定義參數解析器

可以通過實現 HandlerMethodArgumentResolver接口來自定義參數解析邏輯。

@RequestMapping("/custom")
public String custom(CustomObject customObject) {
// 使用自定義對象進行處理
    return "";
}

三、驗證請求參數

請求參數的驗證需要在 Controller層完成,如下代碼,對 nickname進行判空處理,參數驗證一般有 2種方式:

  • 原始方式,這種方式比較靈活,如果需要對參數進行一些邏輯計算后再校驗;
  • 借助三方工具,比如 Spring validation,javax validation等,這種方式靈活度會低一些,但是更優雅;
// 原始方式校驗參數
@RestController
public class UserController {
    @PostMapping("/user/register")
    public void getGradeById(@RequestBody User user) {
        // 代碼邏輯
        if (StringUtils.isBlank(user.getNickname)) {
            throw new Exception("Nickname is required.");
        }
    }
}

或者使用 Spring validation驗證機制,Controller需要增加@Validated注解,User對象中增加@NotBlank注解。

// 借助Spring validation方式校驗參數
@RestController
public class UserController {
    @PostMapping("/user/register")
    public void getGradeById(@Validated @RequestBody User user) {
        // 代碼邏輯
    }
}

public class User {
    @NotBlank(message = "Nickname is required.")
    private String nickname;
    private Integer age;
    // getters and setters and constructors
}

四、調用業務方法

如下代碼,調用 UserService.register()進行注冊業務處理:

@RestController
public class UserController {
    private final UserService userService;
    public UserController(UserService userService) {
        this.userService = userService;
    }

    @PostMapping("/user/register")
    public void getGradeById(@Validated @RequestBody User user) {
        // 調用注冊的業務方法
        userService.register(user);
    }
}

public class User {
    @NotBlank(message = "Nickname is required.")
    private String nickname;
    private Integer age;
    // getters and setters and constructors
}

關于調用業務方法,這里的業務方法是寫一個大而全的方法?還是需要按業務歸類?

遵守一個原則:有強關聯性的邏輯放在一個service方法內,沒有強關聯性的單令拎出來。

這里以用戶注冊之后需要新人發券為例進行說明:

大而全的方法:

    @PostMapping("/user/register")
    public void getGradeById(@Validated @RequestBody User user) {
        // 調用注冊的業務方法
        userService.doRegister(user);
    }
    
    public String doRegister(Uswr user){
        String userId = userService.register(user);
        coupon.sendCoupon(userId);
        // 其他業務邏輯
        return userId;
    }

業務歸類:

    @PostMapping("/user/register")
    public void getGradeById(@Validated @RequestBody User user) {
        // 調用注冊的業務方法
        userService.register(user);
        coupon.sendCoupon(userId);
    }

五、組織返回數據

如下代碼,調用 UserService.register()進行注冊業務處理:

@RestController
public class UserController {
    private final UserService userService;

    public UserController(UserService userService) {
        this.userService = userService;

    }

    @PostMapping("/user/register")
    public UserResponse getGradeById(@Validated @RequestBody User user) {
        // 調用注冊的業務方法
        String userId = userService.regist(user);

        return new UserResponse(userId, user.getNickname);
    }
}

public class UserResponse {
    private String userId;
    private String nickname;
    // getters and setters and constructors
}

六、統一異常處理

比如上述過程在 userService.regist(user);出現異常時,可以做一個try-catch,然后在 Controller層封裝有業務意思的異常信息:

@RestController
public class UserController {
    private final UserService userService;

    @PostMapping("/user/register")
    public UserResponse getGradeById(@Validated @RequestBody User user) {
        // 調用注冊的業務方法
        try {
            String userId = userService.regist(user);
        } catch (Exception e) {
            throw new CustomException();
        }
        return new UserResponse(userId, user.getNickname);
    }
}

建議和總結

看過很多代碼,業務邏輯全部寫在 Controller層,并不能說這樣的做法是錯的,但是看起來很別扭,不優雅!因此,建議在編寫代碼時,最好能遵守一個比較好的規范,比如常見的SOLID規范。

SOLID 實際上是五個設計原則首字母的縮寫,它們分別是:

  • 單一職責原則(Single responsibility principle, SRP)
  • 開放封閉原則(Open–closed principle, OCP)
  • Liskov 替換原則(Liskov substitution principle, LSP)
  • 接口隔離原則(Interface segregation principle, ISP)
  • 依賴倒置原則(Dependency inversion principle, DIP)

另外,建議我們技術人員平時多去閱讀一些優秀開源框架,學習他們的設計思想,代碼規范,相信我:養成一個良好的編碼規范,絕對受益頗多!

責任編輯:趙寧寧 來源: 猿java
相關推薦

2024-08-16 21:38:43

Python代碼編程

2019-04-19 08:04:57

程序員Dockerfile容器

2024-06-17 08:04:23

2012-07-11 10:51:37

編程

2024-03-08 08:00:00

Python開發裝飾器

2024-05-11 18:48:40

技巧代碼技能

2022-05-24 06:07:48

JShack用戶代碼

2022-08-01 23:45:23

代碼識別項目

2024-10-14 08:46:50

Controller開發代碼

2022-06-07 09:30:35

JavaScript變量名參數

2021-01-04 07:57:07

C++工具代碼

2021-03-17 08:00:59

JS語言Javascript

2019-09-20 15:47:24

代碼JavaScript副作用

2022-03-11 12:14:43

CSS代碼前端

2023-07-30 22:25:00

JavaScrip服務端Web

2024-03-20 08:00:00

軟件開發Java編程語言

2023-10-10 08:00:00

2015-01-28 14:30:31

android代碼

2022-06-27 06:23:23

代碼編程

2021-06-08 09:35:11

Cleaner ReaReact開發React代碼
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 91精品国产91久久久久久吃药 | 一区二区精品在线 | 国产欧美在线播放 | 欧美不卡一区二区 | 麻豆毛片 | 亚洲在线一区二区三区 | 最新超碰 | 国产在线播 | 在线欧美一区 | 国产精品69毛片高清亚洲 | 久久久久久综合 | 91免费在线| 玖玖爱365| 日本激情视频在线播放 | 精品欧美一区二区三区久久久 | 日韩欧美不卡 | 天天干天天谢 | 日韩精品在线一区 | 成年人在线观看视频 | 亚洲欧美日韩精品久久亚洲区 | 久久久噜噜噜久久中文字幕色伊伊 | 欧美精品在线免费 | 亚洲国产精品久久久久 | 日韩精品一区二区三区 | 国产精品毛片一区二区三区 | 精品久久久网站 | 成人一区二区三区在线观看 | 国产九九精品视频 | 精品久久久久香蕉网 | 国产精品久久久久久久久久久久久 | 欧美一级大片 | 国产玖玖 | 国产一级在线 | 欧美精品久久久久 | 91久久精品国产91久久性色tv | 一级全黄视频 | 一区二区三区免费 | 国产欧美精品一区二区 | www亚洲一区 | 久久免费精品视频 | 九九视频在线观看视频6 |