偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

Spring Boot+原生注解@JsonView 輕松過濾字段,真的優(yōu)雅!

開發(fā) 前端
@JsonView 是 Jackson 庫(kù)提供的一個(gè)注解,Spring Boot 對(duì)它有原生支持。簡(jiǎn)單來說,它就像一個(gè) “數(shù)據(jù)篩子”,可以在序列化(把 Java 對(duì)象轉(zhuǎn)成 JSON)時(shí),根據(jù)不同的場(chǎng)景決定哪些字段要展示,哪些要隱藏。

兄弟們,今天咱們來聊聊 Spring Boot 里一個(gè)堪稱 “數(shù)據(jù)化妝師” 的神器 ——@JsonView。想象一下,你開發(fā)了一個(gè)接口,前端說:“我只要用戶的姓名和郵箱,別給我密碼和身份證號(hào)!” 這時(shí)候,你是不是習(xí)慣性地想寫一堆 DTO?或者用 @JsonIgnore 一頓亂標(biāo)?別急,@JsonView 能讓你用更優(yōu)雅的姿勢(shì)解決這個(gè)問題。

一、@JsonView 是什么?能吃嗎?

@JsonView 是 Jackson 庫(kù)提供的一個(gè)注解,Spring Boot 對(duì)它有原生支持。簡(jiǎn)單來說,它就像一個(gè) “數(shù)據(jù)篩子”,可以在序列化(把 Java 對(duì)象轉(zhuǎn)成 JSON)時(shí),根據(jù)不同的場(chǎng)景決定哪些字段要展示,哪些要隱藏。比如:

  • 用戶注冊(cè)接口:只返回用戶名和郵箱。
  • 用戶詳情接口:返回所有字段,包括地址和手機(jī)號(hào)。
  • 管理員接口:甚至可以返回敏感信息(但記得加密哦?。?。

它的核心思想是視圖(View)。你可以定義多個(gè)視圖接口,每個(gè)接口代表一種數(shù)據(jù)展示規(guī)則。然后在實(shí)體類的字段上標(biāo)注這些視圖,最后在控制器方法里指定用哪個(gè)視圖。就這么簡(jiǎn)單!

二、入門案例:給用戶數(shù)據(jù)化個(gè)淡妝

咱們先來看一個(gè)簡(jiǎn)單的例子。假設(shè)我們有一個(gè) User 類:

public class User {
    @JsonView(User.BaseView.class)
    private Long id;
    @JsonView(User.BaseView.class)
    private String username;
    @JsonView(User.DetailView.class)
    private String email;
    @JsonView(User.AdminView.class)
    private String password;
    // 視圖接口定義
    public interface BaseView {}
    public interface DetailView extends BaseView {}
    public interface AdminView extends DetailView {}
}

這里定義了三個(gè)視圖:

  • BaseView:基礎(chǔ)信息,包含 id 和 username。
  • DetailView:繼承自 BaseView,額外包含 email。
  • AdminView:繼承自 DetailView,額外包含 password。

接下來,在控制器里指定視圖:

@RestController
@RequestMapping("/users")
public class UserController {
    @GetMapping("/{id}")
    @JsonView(User.BaseView.class)
    public User getUser(@PathVariable Long id) {
        // 假設(shè)這里從數(shù)據(jù)庫(kù)查詢用戶
        return userService.findById(id);
    }
    @GetMapping("/detail/{id}")
    @JsonView(User.DetailView.class)
    public User getDetailUser(@PathVariable Long id) {
        return userService.findById(id);
    }
    @GetMapping("/admin/{id}")
    @JsonView(User.AdminView.class)
    public User getAdminUser(@PathVariable Long id) {
        return userService.findById(id);
    }
}

這樣,三個(gè)接口就會(huì)返回不同的字段:

  • /users/1:返回{"id":1, "username":"張三"}。
  • /users/detail/1:返回{"id":1, "username":"張三", "email":"zhangsan@example.com"}。
  • /users/admin/1:返回所有字段,包括password(但實(shí)際項(xiàng)目中記得加密?。?。

是不是比寫三個(gè) DTO 清爽多了?而且視圖接口可以無限繼承,靈活組合。比如,如果某個(gè)接口需要同時(shí)展示 BaseView 和 DetailView 的字段,你可以再定義一個(gè)復(fù)合視圖:

public interface CompositeView extends BaseView, DetailView {}

然后在控制器方法上用@JsonView(CompositeView.class),就這么簡(jiǎn)單!

三、進(jìn)階玩法:處理關(guān)聯(lián)對(duì)象的千層餅

實(shí)際項(xiàng)目中,對(duì)象往往不是孤立的。比如,一個(gè) User 可能關(guān)聯(lián)一個(gè) Order,Order 又關(guān)聯(lián)一個(gè) Product。這時(shí)候,@JsonView 的嵌套處理就顯得尤為重要。

3.1 簡(jiǎn)單關(guān)聯(lián):返回空殼對(duì)象

假設(shè) User 有一個(gè) Order 字段:

public class User {
    // ...其他字段
    @JsonView(User.OrderView.class)
    private Order order;
    public interface OrderView {}
}
public class Order {
    @JsonView(Order.BaseView.class)
    private Long id;
    @JsonView(Order.DetailView.class)
    private Date createTime;
    public interface BaseView {}
    public interface DetailView extends BaseView {}
}

如果在控制器中使用@JsonView(User.OrderView.class),返回的 JSON 會(huì)是:

{
    "id": 1,
    "username": "張三",
    "order": {}
}

注意,這里的 order 是一個(gè)空對(duì)象。因?yàn)?User 的 OrderView 只標(biāo)記了 order 字段本身,而 Order 類的字段沒有被當(dāng)前視圖覆蓋。這時(shí)候,Jackson 會(huì)默認(rèn)返回空對(duì)象,而不是遞歸序列化所有字段。

3.2 深度關(guān)聯(lián):繼承視圖解千層

如果我們希望返回 Order 的 id 和 createTime,該怎么辦呢?很簡(jiǎn)單,讓 User 的 OrderView 繼承 Order 的 BaseView:

public class User {
    // ...其他字段
    @JsonView(User.OrderView.class)
    private Order order;
    public interface OrderView extends Order.BaseView {}
}
public class Order {
    @JsonView(Order.BaseView.class)
    private Long id;
    @JsonView(Order.DetailView.class)
    private Date createTime;
    public interface BaseView {}
    public interface DetailView extends BaseView {}
}

然后在控制器中使用@JsonView(User.OrderView.class),返回的 JSON 就會(huì)是:

{
    "id": 1,
    "username": "張三",
    "order": {
        "id": 1001,
        "createTime": "2023-10-01T12:00:00"
    }
}

這里的關(guān)鍵是視圖繼承。User.OrderView 繼承了 Order.BaseView,所以 Jackson 在序列化 order 字段時(shí),會(huì)應(yīng)用 Order.BaseView 的規(guī)則,即包含 id 字段。如果還需要 createTime,可以讓 User.OrderView 繼承 Order.DetailView:

public interface OrderView extends Order.DetailView {}

這樣,返回的 JSON 就會(huì)包含 createTime 字段。

3.3 多層嵌套:鏈?zhǔn)嚼^承無壓力

如果 Order 還關(guān)聯(lián)了 Product,Product 又關(guān)聯(lián)了 Category,該怎么辦呢?別慌,繼續(xù)用繼承:

public class Order {
    // ...其他字段

    @JsonView(Order.ProductView.class)
    private Product product;

    publicinterface ProductView extends Product.BaseView {}
}

publicclass Product {
    @JsonView(Product.BaseView.class)
    private Long id;

    @JsonView(Product.DetailView.class)
    private String name;

    @JsonView(Product.CategoryView.class)
    private Category category;

    publicinterface BaseView {}
    publicinterface DetailView extends BaseView {}
    publicinterface CategoryView extends Category.BaseView {}
}

publicclass Category {
    @JsonView(Category.BaseView.class)
    private Long id;

    @JsonView(Category.DetailView.class)
    private String name;

    publicinterface BaseView {}
    publicinterface DetailView extends BaseView {}
}

然后在控制器中使用@JsonView(User.OrderView.class),其中 User.OrderView 繼承了 Order.ProductView,而 Order.ProductView 又繼承了 Product.CategoryView,最終會(huì)返回:

{
    "id": 1,
    "username": "張三",
    "order": {
        "id": 1001,
        "createTime": "2023-10-01T12:00:00",
        "product": {
            "id": 2001,
            "category": {
                "id": 3001
            }
        }
    }
}

這樣,通過層層繼承,我們可以靈活控制任意深度的嵌套對(duì)象序列化。

四、動(dòng)態(tài)視圖:讓數(shù)據(jù)展示更靈活

前面的例子都是在控制器方法上用 @JsonView 注解指定視圖,這種方式適合固定場(chǎng)景。但如果我們需要根據(jù)用戶角色、請(qǐng)求參數(shù)等動(dòng)態(tài)決定視圖,該怎么辦呢?這時(shí)候,可以使用MappingJacksonValue類。

4.1 基于用戶角色的動(dòng)態(tài)視圖

假設(shè)我們有一個(gè)接口,普通用戶只能看到 BaseView,管理員可以看到 AdminView。我們可以這樣做:

@GetMapping("/dynamic/{id}")
public MappingJacksonValue getDynamicUser(@PathVariable Long id) {
    User user = userService.findById(id);
    MappingJacksonValue mapping = new MappingJacksonValue(user);

    // 獲取當(dāng)前用戶角色
    Authentication authentication = SecurityContextHolder.getContext().getAuthentication();
    if (authentication != null && authentication.getAuthorities().contains(new SimpleGrantedAuthority("ROLE_ADMIN"))) {
        mapping.setSerializationView(User.AdminView.class);
    } else {
        mapping.setSerializationView(User.BaseView.class);
    }

    return mapping;
}

這樣,管理員訪問時(shí)會(huì)返回所有字段,普通用戶只能看到基礎(chǔ)信息。

4.2 基于請(qǐng)求參數(shù)的動(dòng)態(tài)視圖

如果希望根據(jù)請(qǐng)求參數(shù)(如?view=detail)來決定視圖,可以這樣做:

@GetMapping("/dynamic/{id}")
public MappingJacksonValue getDynamicUser(@PathVariable Long id, @RequestParam(defaultValue = "base") String view) {
    User user = userService.findById(id);
    MappingJacksonValue mapping = new MappingJacksonValue(user);

    switch (view) {
        case"detail":
            mapping.setSerializationView(User.DetailView.class);
            break;
        case"admin":
            mapping.setSerializationView(User.AdminView.class);
            break;
        default:
            mapping.setSerializationView(User.BaseView.class);
    }

    return mapping;
}

這樣,前端可以通過參數(shù)靈活選擇需要的視圖。

五、性能優(yōu)化:別讓 @JsonView 拖后腿

雖然 @JsonView 很方便,但如果使用不當(dāng),可能會(huì)影響性能。比如,當(dāng)處理大量數(shù)據(jù)時(shí),頻繁的反射和視圖解析可能會(huì)帶來額外開銷。不過,通過合理設(shè)計(jì),我們可以將性能影響降到最低。

5.1 避免過度使用視圖繼承

視圖繼承雖然靈活,但如果嵌套層次過深,可能會(huì)導(dǎo)致 Jackson 在序列化時(shí)進(jìn)行大量的類檢查。建議將視圖繼承控制在合理范圍內(nèi),或者使用復(fù)合視圖代替多層繼承。

5.2 緩存視圖信息

在高并發(fā)場(chǎng)景下,可以考慮緩存視圖信息。例如,將視圖類和字段的映射關(guān)系緩存到 ConcurrentHashMap 中,避免每次序列化都反射解析字段。

5.3 與 DTO 結(jié)合使用

對(duì)于極其復(fù)雜的場(chǎng)景,@JsonView 可能會(huì)讓實(shí)體類變得臃腫。這時(shí)候,可以結(jié)合 DTO 使用:用 @JsonView 處理簡(jiǎn)單場(chǎng)景,用 DTO 處理復(fù)雜的數(shù)據(jù)轉(zhuǎn)換。這樣既能保持代碼簡(jiǎn)潔,又能提升性能。

六、常見問題及解決方案

6.1 關(guān)聯(lián)對(duì)象返回空殼

問題:當(dāng)使用 @JsonView 處理關(guān)聯(lián)對(duì)象時(shí),返回的是一個(gè)空對(duì)象,而不是期望的字段。

解決方案:確保關(guān)聯(lián)對(duì)象的字段被當(dāng)前視圖覆蓋??梢酝ㄟ^視圖繼承或直接在關(guān)聯(lián)對(duì)象的字段上標(biāo)注當(dāng)前視圖。

6.2 視圖接口無法繼承

問題:在 Java 8 中,接口不能有默認(rèn)方法,導(dǎo)致視圖繼承時(shí)無法共享公共字段。

解決方案:使用標(biāo)記接口,或者在父視圖中定義公共字段,子視圖繼承父視圖。

6.3 與 @JsonIgnore 沖突

問題:當(dāng)字段同時(shí)被 @JsonView 和 @JsonIgnore 標(biāo)注時(shí),@JsonView 會(huì)被忽略。

解決方案:@JsonIgnore 的優(yōu)先級(jí)高于 @JsonView。如果需要同時(shí)使用,建議在視圖中排除該字段,而不是使用 @JsonIgnore。

6.4 包裝返回結(jié)果失效

問題:當(dāng)使用統(tǒng)一的 Result 包裝類返回?cái)?shù)據(jù)時(shí),@JsonView 無法過濾字段。

解決方案:在 Result 類中也應(yīng)用 @JsonView,并確保視圖接口正確繼承。或者,使用 ResponseBodyAdvice 攔截響應(yīng),動(dòng)態(tài)設(shè)置視圖。

七、與其他注解的對(duì)比

7.1 @JsonView vs @JsonIgnore

  • @JsonIgnore:簡(jiǎn)單直接,但只能靜態(tài)排除字段,無法根據(jù)場(chǎng)景動(dòng)態(tài)調(diào)整。
  • @JsonView:靈活強(qiáng)大,可以動(dòng)態(tài)控制字段展示,但需要定義視圖接口。

結(jié)論:如果需要?jiǎng)討B(tài)控制字段,優(yōu)先使用 @JsonView;如果是靜態(tài)排除,@JsonIgnore 更簡(jiǎn)單。

7.2 @JsonView vs DTO

  • DTO:清晰直觀,適合復(fù)雜數(shù)據(jù)轉(zhuǎn)換,但會(huì)增加類的數(shù)量。
  • @JsonView:減少類的數(shù)量,保持實(shí)體類簡(jiǎn)潔,但可能使代碼邏輯分散。

結(jié)論:簡(jiǎn)單場(chǎng)景用 @JsonView,復(fù)雜場(chǎng)景用 DTO,或者兩者結(jié)合。

八、最佳實(shí)踐

  1. 視圖命名規(guī)范:視圖接口名稱應(yīng)與業(yè)務(wù)場(chǎng)景一致,如 User.BaseView、Order.DetailView。
  2. 繼承深度控制:避免超過三層繼承,必要時(shí)使用復(fù)合視圖。
  3. 敏感數(shù)據(jù)處理:敏感字段(如密碼)應(yīng)單獨(dú)放在 AdminView 中,并結(jié)合加密處理。
  4. 文檔說明:在代碼注釋中說明每個(gè)視圖的用途,方便團(tuán)隊(duì)成員理解。
  5. 單元測(cè)試:對(duì)每個(gè)視圖接口編寫測(cè)試用例,確保返回字段符合預(yù)期。

九、總結(jié)

@JsonView 是 Spring Boot 中一個(gè)被低估的神器,它讓我們可以用更優(yōu)雅的方式控制 JSON 序列化,避免了大量冗余的 DTO 和注解。通過合理設(shè)計(jì)視圖接口,結(jié)合動(dòng)態(tài)視圖和性能優(yōu)化,我們可以在保證代碼簡(jiǎn)潔的同時(shí),滿足各種復(fù)雜的業(yè)務(wù)需求。

下次遇到 “這個(gè)接口需要返回某些字段,那個(gè)接口不需要” 的需求時(shí),別再寫 DTO 了,試試 @JsonView 吧!它真的能讓你的代碼更優(yōu)雅,更有逼格。

雖然 @JsonView 很強(qiáng)大,但也別濫用。對(duì)于極其復(fù)雜的場(chǎng)景,還是要結(jié)合其他工具(如 MapStruct)來處理。技術(shù)沒有銀彈,合適的才是最好的。

責(zé)任編輯:武曉燕 來源: 石杉的架構(gòu)筆記
相關(guān)推薦

2025-04-10 00:22:22

Spring@JsonView字段

2025-04-08 02:22:22

SpringJackson注解

2023-04-17 23:49:09

開發(fā)代碼Java

2025-04-10 00:25:00

Spring@JsonView注解

2021-10-22 14:50:23

Spring BootJava

2023-06-02 16:24:46

SpringBootSSM

2020-04-23 15:59:04

SpringKafka集群

2025-02-07 09:11:04

JSON對(duì)象策略

2022-09-22 13:28:34

Redis分布式鎖

2022-09-29 08:28:57

SpringRedis分布式

2024-08-09 08:52:26

2025-05-13 07:13:25

2017-08-02 14:44:06

Spring Boot開發(fā)注解

2009-09-27 14:01:29

Spring MVC

2024-08-02 09:15:22

Spring捕捉格式

2025-03-11 00:55:00

Spring停機(jī)安全

2021-04-13 20:24:57

Spring Boot注解spring

2024-10-14 17:18:27

2021-04-20 10:50:38

Spring Boot代碼Java

2024-12-06 09:27:28

點(diǎn)贊
收藏

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