精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

告別異常處理:用Result模式和Discriminated Union打造優雅的C#錯誤處理機制

開發 前端
你還有一個中間件可以捕獲所有異常并向用戶返回適當的響應。但過了一段時間后,你決定通過引入Result類型來重構代碼,以避免使用異?!驗楫惓1緫糜谔厥馇闆r。

假設你有一段根據輸入參數返回不同結果的代碼——這很常見。

有幾種方法可以實現這個需求。為了說明我的意思,假設你有以下模型:

public sealed class Note
{
    public Guid Id { get; set; }

    public string Title { get; set; } = null!;

    public Guid UserId { get; set; }

    public DateTimeOffset CreatedAt { get; set; }

    public DateTimeOffset? UpdatedAt { get; set; }
};

這是一個嘗試更新該模型的代碼示例:

public sealed class UpdateNoteHandler(INoteRepository noteRepository)
{
    public async Task InvokeAsync(
        Guid noteId,
        string? title,
        Guid userId)
    {
        var note = await noteRepository.GetNoteOrNullAsync(noteId);

        if (note == null)
        {
            thrownew InvalidOperationException("Note was not found.");
        }

        if (note.UserId != userId)
        {
            thrownew InvalidOperationException("Forbidden.");
        }

        if (string.IsNullOrWhiteSpace(title))
        {
            thrownew InvalidOperationException("Invalid input.");
        }

        note.Title = title;
        note.UpdatedAt = DateTimeOffset.UtcNow;

        await noteRepository.UpdateNoteAsync(note);
    }
}

這個處理程序會為類中定義的每個錯誤拋出異常。

你還有一個中間件可以捕獲所有異常并向用戶返回適當的響應。

但過了一段時間后,你決定通過引入Result類型來重構代碼,以避免使用異常——因為異常本應用于特殊情況。

假設你創建了這樣的東西:

public sealed class Result
{
    public bool IsSuccess => string.IsNullOrWhiteSpace(Error);
    
    public string? Error { get; set; }
}

之后,你將方法更新為這樣:

public sealed class UpdateNoteHandler(INoteRepository noteRepository)
{
    public async Task<Result> InvokeAsync(
        Guid noteId,
        string? title,
        Guid userId)
    {
        var note = await noteRepository.GetNoteOrNullAsync(noteId);

        if (note == null)
        {
            returnnew Result
            {
                Error = "Note was not found."
            };
        }

        if (note.UserId != userId)
        {
            returnnew Result
            {
                Error = "Forbidden."
            };
        }

        if (string.IsNullOrWhiteSpace(title))
        {
            returnnew Result
            {
                Error = "Invalid input."
            };
        }

        note.Title = title;
        note.UpdatedAt = DateTimeOffset.UtcNow;

        await noteRepository.UpdateNoteAsync(note);

        returnnew Result();
    }
}

但目前很難理解每個錯誤代表什么。因此,你希望根據錯誤類型對Result進行分類。

為此,你可能會引入一個ErrorType并像這樣更新Result:

public enum ErrorType
{
    NotFound,
    Forbidden,
    InvalidInput
}

publicsealedclassResult
{
    publicbool IsSuccess => string.IsNullOrWhiteSpace(Error);

    publicstring? Error { get; set; }

    public ErrorType? ErrorType { get; set; }
}

然后,你將方法更新為這樣:

public sealed class UpdateNoteHandler(INoteRepository noteRepository)
{
    public async Task<Result> InvokeAsync(
    Guid noteId,
    string? title,
    Guid userId)
    {
        var note = await noteRepository.GetNoteOrNullAsync(noteId);

        if (note == null)
        {
            returnnew Result
            {
                Error = "Note was not found.",
                ErrorType = ErrorType.NotFound
            };
        }

        if (note.UserId != userId)
        {
            returnnew Result
            {
                Error = "Forbidden.",
                ErrorType = ErrorType.Forbidden
            };
        }

        if (string.IsNullOrWhiteSpace(title))
        {
            returnnew Result
            {
                Error = "Invalid input.",
                ErrorType = ErrorType.InvalidInput
            };
        }

        note.Title = title;
        note.UpdatedAt = DateTimeOffset.UtcNow;

        await noteRepository.UpdateNoteAsync(note);

        returnnew Result();
    }
}

這個方法有效——但只到一定程度。

假設你想為筆記未找到的情況添加一個額外屬性。

但該怎么做?你應該在通用的Result中引入一個新屬性嗎?

像這樣:

public sealed class Result
{
    public bool IsSuccess => string.IsNullOrWhiteSpace(Error);

    public string? Error { get; set; }

    public ErrorType? ErrorType { get; set; }

    public Guid? NoteId { get; set; } // <-- 用于附加數據的新屬性
}

嗯...如果需要添加更多額外數據呢?

這時Result會變得混亂且難以使用——你現在還必須檢查額外數據。

可能會變成這樣:

var result = await hanlder.InvokeAsync();

if(!result.IsSuccess &&
   result.ErrorType == ErrorType.NotFound &&
   result.NoteId.HasValue)
{
    return Results.NoteFound($"There's not such note with id `{result.NoteId.Value}`");
}

這相當煩人——但我很高興告訴你有一個更好的方法來改進這段代碼。

首先,讓我們創建一個Reply類——它將作為我們處理程序結果的基類。

public class Reply;

現在,讓我們為處理程序中的每種情況引入特定的Reply類型:

public sealed class NotFoundReply(Guid noteId) : Reply
{
    public Guid NoteId { get; } = noteId;
}

publicsealedclassForbiddenReply : Reply;

publicsealedclassEmptyTitleReply : Reply;

publicsealedclassSuccessReply : Reply;

如你所見,你的類中不再需要ErrorType枚舉或Error屬性——類型本身已經告訴你服務返回了哪種回復,這非???。

更棒的是,你可以只為特定情況擴展回復類所需的數據——就像NotFoundReply所做的那樣。

很酷,不是嗎?

但讓我們回到我們離開的地方。

現在我要更新我們的處理程序——它將看起來像這樣:

public sealed class UpdateNoteHandler(INoteRepository noteRepository)
{
    public async Task<Reply> InvokeAsync(
        Guid noteId,
        string? title,
        Guid userId)
    {
        var note = await noteRepository.GetNoteOrNullAsync(noteId);

        if (note == null)
        {
            returnnew NotFoundReply(noteId);
        }

        if (note.UserId != userId)
        {
            returnnew ForbiddenReply();
        }

        if (string.IsNullOrWhiteSpace(title))
        {
            returnnew EmptyTitleReply();
        }

        note.Title = title;
        note.UpdatedAt = DateTimeOffset.UtcNow;

        await noteRepository.UpdateNoteAsync(note);

        returnnew SuccessReply();
    }
}

這已經比之前使用Result的方法好多了——但我們還能做得更好。

更好?有什么能比多態方法更好?它有什么問題?

問題在于使用多態方法時,我們無法真正控制具體的類型。任何人都可以輕松添加Reply的新子類,而忘記在代碼中的某處正確處理它。

為了避免這種情況,讓我們引入一個Discriminated Union(可區分聯合),并將我們的Reply重構為DU。

public abstractclassUpdateNoteReply
{
    private UpdateNoeReply()
    {
    }  

    public sealed class NotFoundReply(Guid noteId) : UpdateNoeReply
    {
        public Guid NoteId { get; } = noteId;
    }

    publicsealedclassForbidden : UpdateNoteReply;

    publicsealedclassEmptyTitle : UpdateNoteReply;

    publicsealedclassSuccess : UpdateNoteReply;
}

現在我們獲得了使用Discriminated Union的一些優勢:

? 更容易推理所有可能的狀態

? 設計上就是不可變的

現在你的處理程序將如下所示:

public sealed class UpdateNoteHandler4(INoteRepository noteRepository)
{
    public async Task<UpdateNoteReply> InvokeAsync(
        Guid noteId,
        string? title,
        Guid userId)
    {
        var note = await noteRepository.GetNoteOrNullAsync(noteId);

        if (note == null)
        {
            returnnew UpdateNoteReply.NotFound(noteId);
        }

        if (note.UserId != userId)
        {
            returnnew  UpdateNoteReply.Forbidden();
        }

        if (string.IsNullOrWhiteSpace(title))
        {
            returnnew UpdateNoteReply.EmptyTitle();
        }

        note.Title = title;
        note.UpdatedAt = DateTimeOffset.UtcNow;

        await noteRepository.UpdateNoteAsync(note);

        returnnew UpdateNoteReply.Success();
    }
}

之后,你也可以輕松更新你的端點:

app.MapPost("notes/{id:guid}", async (
    [FromServices] UpdateNoteHandler handler,
    [FromQuery] Guid id) =>
{
    var reply = await handler.InvokeAsync(id, title: "", userId: Guid.Empty);

    return reply switch
    {
        UpdateNoteReply.NotFound notFound => Results.NotFound(notFound.NoteId),
        UpdateNoteReply.EmptyTitle => Results.BadRequest(),
        UpdateNoteReply.Forbidden => Results.Forbid(),
        _ => Results.Ok()
    };
});

通過使用Discriminated Union,我們獲得了:

? 干凈可讀的回復結構

? 能夠使用switch語句

? 僅在真正需要的地方添加額外數據

你怎么看?你在服務中如何返回不同的回復?

責任編輯:武曉燕 來源: 架構師老盧
相關推薦

2021-03-02 09:12:25

Java異常機制

2009-08-05 18:09:17

C#異常處理機制

2009-06-19 16:20:14

ASP.NET錯誤處理

2009-07-31 11:28:42

錯誤處理機制ASP.NET

2011-03-17 09:20:05

異常處理機制

2025-02-10 09:49:00

2009-09-02 18:34:28

C#鼠標事件

2024-03-04 10:00:35

數據庫處理機制

2011-04-06 10:27:46

Java異常處理

2011-07-21 15:20:41

java異常處理機制

2015-12-28 11:25:51

C++異常處理機制

2015-10-09 13:54:14

切面編程錯誤處理機制

2024-04-16 12:18:05

編程異常處理錯誤返回

2021-07-03 17:53:52

Java異常處理機制

2020-12-17 06:25:05

Gopanic 模式

2010-03-05 15:40:16

Python異常

2023-03-08 08:54:59

SpringMVCJava

2021-03-25 07:44:39

C++異常處理開發技術

2021-04-14 07:08:14

Nodejs錯誤處理

2009-08-05 18:28:05

C#異常處理
點贊
收藏

51CTO技術棧公眾號

女生裸体视频一区二区三区| 黄页网站大全在线免费观看| 日韩精品国产欧美| 在线视频日本亚洲性| 久久人人爽av| 99在线视频影院| 国产亚洲一区二区三区在线观看| 国产精品偷伦视频免费观看国产 | 亚洲男女性事视频| 污网站免费在线| sqte在线播放| 国产精品久久久久久久久快鸭| 超碰97人人在线| 久草视频在线免费| 欧美精品aa| 国产一区二区三区在线看 | 国产精品 欧美 日韩| 国产亚洲一区二区三区啪| 51午夜精品国产| 国产精品不卡视频| 成人做爰66片免费看网站| 波多野结衣啪啪| 欧美国产高清| 色阁综合伊人av| 中文字幕在线免费看线人| 美女精品久久| 欧美午夜电影网| 久久久久久久中文| 一二三四区在线观看| 国产片一区二区| 国内一区在线| 国产手机av在线| 日韩精品一区第一页| 久久久久久久91| 女同久久另类69精品国产| 久久av资源| 精品国产乱码久久久久久图片 | 都市激情亚洲综合| 亚洲国产一区二区三区| 麻豆md0077饥渴少妇| 91大神在线网站| www亚洲一区| 精品久久久久久一区二区里番| 国产高清免费观看| 精品一区二区日韩| 国产精品一久久香蕉国产线看观看| 欧美一区二区激情视频| 亚洲高清电影| 欧美激情乱人伦| 中文字幕av免费在线观看| 国产精品二区不卡| 日韩在线观看你懂的| 东方伊人免费在线观看| 精品久久美女| 正在播放国产一区| 亚洲天堂岛国片| 日韩精品免费一区二区在线观看 | 中文字幕人妻一区二区| 日韩精品a在线观看91| 亚洲国产精品人久久电影| 在线观看免费视频国产| 精品国产午夜肉伦伦影院| 亚洲白拍色综合图区| 色悠悠在线视频| 凹凸成人在线| 亚洲精品国产精品国自产在线| 亚洲最大免费视频| 亚洲影院天堂中文av色| 亚洲深夜福利在线| 日本一区二区视频在线播放| 久久综合成人| 欧美成人在线免费| 国产一级aa大片毛片| 9色精品在线| 欧美一区二区三区……| 久草热在线观看| 久草中文综合在线| 97影院在线午夜| 香蕉视频黄色片| 中文字幕乱码日本亚洲一区二区| 亚洲五月六月| 日本一级理论片在线大全| 精品欧美激情精品一区| 成年人视频网站免费观看| 精品成人免费一区二区在线播放| 欧美视频第二页| 久久发布国产伦子伦精品| 久久精品福利| 中文在线不卡视频| 欧美黑吊大战白妞| 国产精品久久久免费 | 国产精品自拍合集| 欧美aa免费在线| 欧美日韩一级片网站| 韩国一区二区三区四区| 妖精视频一区二区三区免费观看| 国产精品一区二区在线观看网站| 精品久久久久久中文字幕一区奶水 | 亚洲视频 欧洲视频| 热久久最新网址| 狼人综合视频| 91精品国模一区二区三区| 国产51自产区| 四虎成人av| 午夜精品福利视频| 中文字幕一区二区久久人妻| 国产98色在线|日韩| 日韩国产欧美精品| 黑人极品ⅴideos精品欧美棵| 在线一区二区三区| 性活交片大全免费看| 不卡在线一区二区| 久久久免费精品| 亚洲综合精品视频| www成人在线观看| 一二三在线视频| 欧美日韩五区| 亚洲精品白浆高清久久久久久| 国产乱子轮xxx农村| 国产精品免费看| 亚洲综合在线做性| 在线视频婷婷| 精品久久中文字幕久久av| 日韩av片免费观看| 精品理论电影在线| 国产69精品久久久久99| 99国产精品一区二区三区 | 日韩午夜av一区| 99在线视频免费| 欧美一级一区| 精品日韩电影| av手机在线观看| 日韩午夜小视频| 男人在线观看视频| 免费在线观看成人| 日本一区二区在线视频| 国产精品一区二区av影院萌芽| 欧美mv日韩mv亚洲| 麻豆明星ai换脸视频| 久久99精品国产.久久久久| 欧美资源一区| 日韩一区二区三区在线免费观看 | 高清国产一区| 影音先锋中文在线视频| 91精品国产日韩91久久久久久| 国产欧美一区二区三区在线观看视频 | 亚洲天堂1区| 精品久久久久久久人人人人传媒| 久久久久久视频| 国产一区二区在线视频| 中文一区一区三区免费| 久久久免费人体| 综合国产在线视频| 中文字幕在线观看免费| 国产精品色婷婷久久58| 日韩一级免费片| 99久久亚洲精品| 成人午夜黄色影院| a视频在线观看| 精品女同一区二区| 中文字幕第28页| www.亚洲人| 国产中文字幕免费观看| 亚洲动漫精品| 国产精品com| 日韩伦理在线电影| 91麻豆精品国产自产在线| 欧美一级片在线视频| 国产精品综合二区| 被灌满精子的波多野结衣| 噜噜噜天天躁狠狠躁夜夜精品| 2020欧美日韩在线视频| 国产1区2区3区在线| 欧美三级中文字| 日本在线一级片| 成人午夜激情影院| 欧美 日韩精品| 欧美色女视频| 97人人模人人爽人人少妇| 国产精品电影| 最新69国产成人精品视频免费| 一区二区国产欧美| 亚洲一级在线观看| 色爱av美腿丝袜综合粉嫩av | 欧美日韩一视频区二区| 国产一区二区免费在线观看| **欧美日韩在线观看| 久久精品国产欧美亚洲人人爽| 朝桐光av在线一区二区三区| 精品国产成人av| 精品伦精品一区二区三区视频密桃 | 亚洲精品福利视频网站| 天天插天天射天天干| 免费精品99久久国产综合精品| 2022中文字幕| 久久99国产成人小视频| 91精品久久久久久久久久久| 91破解版在线观看| 一本一本久久a久久精品牛牛影视| 国产成人av免费看| 色综合一个色综合亚洲| 青青青在线免费观看| 26uuu国产日韩综合| 两女双腿交缠激烈磨豆腐| 欧美一级视频| 91视频 - 88av| 日韩伦理视频| 久久99九九| 日韩精品一区二区三区中文字幕| 欧美一区二区三区免费观看 | 亚洲黄色免费网站| www.av欧美| 成人爱爱电影网址| 亚洲综合激情视频| 亚洲欧美激情诱惑| 国产精品视频网站在线观看| 日韩aaaa| 蜜桃久久影院| 国产一区丝袜| 99免费在线视频观看| 国产成人毛片| 国产v综合ⅴ日韩v欧美大片| 成人在线高清免费| 欧美xxxx做受欧美| 色网站免费在线观看| 亚洲精品自在久久| 欧美一级在线免费观看| 欧美在线三级| 亚洲天堂av在线免费观看| 亚洲精品18p| 91精品国产欧美一区二区18| 真实的国产乱xxxx在线91| 欧美日韩国产在线| 国产在线综合网| 亚洲欧美激情插| 欧美一级片在线视频| 国产精品视频在线看| 免费毛片视频网站| 久久老女人爱爱| 国产精品无码在线| 91丨porny丨最新| 粉嫩av懂色av蜜臀av分享| av资源网一区| 国产性生活毛片| 成人av在线看| 久久久高清视频| 9i在线看片成人免费| 白丝校花扒腿让我c| 国产成人自拍在线| 永久看看免费大片| 国产成人免费高清| 亚洲成人精品在线播放| 粉嫩av一区二区三区| 国偷自产av一区二区三区麻豆| 国产精品一二三| 日本成人在线免费| 床上的激情91.| 制服丝袜第一页在线观看| 成人黄色在线看| 性欧美丰满熟妇xxxx性久久久| 91丨porny丨首页| 一级黄色片网址| 综合激情成人伊人| 久久久久久天堂| 午夜欧美视频在线观看| 日韩在线视频免费播放| 日本韩国欧美一区| 久久久久久久国产| 国产在线色视频| 日韩中文字幕欧美| 在线看福利影| 午夜精品一区二区三区在线视| 免费h在线看| 国产精品国内视频| 国产美女亚洲精品7777| 99久久伊人精品影院| 日韩影视高清在线观看| 日韩资源av在线| 香蕉久久网站| 欧美日韩精品在线一区二区 | 亚洲图区在线| 一区二区不卡在线视频 午夜欧美不卡'| 婷婷久久综合| 国产精品又粗又长| 日本aⅴ亚洲精品中文乱码| 色婷婷综合在线观看| av一区二区三区四区| 日韩欧美黄色网址| 一区二区三区美女| 成年人av网站| 欧美一区二区三区成人| 亚洲色图 校园春色| 精品国产欧美成人夜夜嗨| 操喷在线视频| 国产精品直播网红| 精品视频自拍| 一区中文字幕在线观看| 亚洲激情亚洲| www.久久av.com| 99久久综合99久久综合网站| 女同久久另类69精品国产| 亚洲成精国产精品女| 中文字幕乱码在线观看| 亚洲高清福利视频| 国产乱色在线观看| 国产成人涩涩涩视频在线观看 | 欧美成人免费网站| 国产午夜在线视频| 久久久久在线观看| 91精品国产色综合久久不卡粉嫩| 精品视频在线观看| 欧美激情性爽国产精品17p| 爱情岛论坛成人| www.亚洲免费av| frxxee中国xxx麻豆hd| 色婷婷亚洲婷婷| av影片在线播放| 99久久人爽人人添人人澡 | av免费一区二区| 久久久99久久精品欧美| 国产午夜激情视频| 欧美一区二区三区精品| 国产青青草在线| 欧洲s码亚洲m码精品一区| 在线视频亚洲欧美中文| 在线电影看在线一区二区三区| 久久伊人亚洲| aa片在线观看视频在线播放| 亚洲一级二级三级在线免费观看| 国产又粗又猛又黄又爽无遮挡 | 在线黄色网页| 91久久精品在线| 欧美激情黄色片| 亚洲 国产 图片| 中文字幕精品三区| 国产精品第六页| 亚洲视屏在线播放| 欧美性猛交xxx高清大费中文| 国产一区二区无遮挡| 国内精品久久久久久久影视蜜臀| 亚洲国产综合av| 亚洲蜜臀av乱码久久精品| 国产精品久久久久久69| 久久精品国产久精国产思思| 欧美国产视频| 在线一区亚洲| 狠狠色狠狠色综合系列| 性生交大片免费全黄| 欧美精品久久久久久久多人混战 | 欧美日韩123| 国产日产欧美视频| 久久丝袜美腿综合| 波多野结衣一区二区在线| 国产一区二区成人| 日韩欧美2区| 一区二区三区四区五区精品| 精品一区二区三区在线观看| 亚洲一级生活片| 日韩欧美区一区二| 丁香花视频在线观看| 韩国一区二区三区美女美女秀| 一本久道久久综合婷婷鲸鱼| 国产精品久久不卡| 91官网在线免费观看| 在线观看av的网站| 91在线精品视频| 亚洲精品国产日韩| 午夜理伦三级做爰电影| 欧美日韩一区二区在线视频| 黄色网址在线免费| 国产女主播一区二区| 久久精品动漫| 亚洲精品电影院| 欧美zozozo| 精品91久久| 自拍偷拍99| av在线播放不卡| 国产精品无码一区| 欧美成人免费一级人片100| 久久九九热re6这里有精品| 欧美三级午夜理伦三级| 国产精品不卡一区二区三区| 亚洲精品久久久久久久久久| 日本精品中文字幕| **女人18毛片一区二区| www国产视频| 欧美三级电影精品| 麻豆蜜桃在线| 日韩久久久久久久| 国产成人精品亚洲日本在线桃色| 国产精品久久久久久久久久久久久久久久久 | 亚洲国内欧美| 亚洲天堂av中文字幕| 亚洲精品一区在线观看| 69堂免费精品视频在线播放| 国产成人亚洲综合无码| 国产三级欧美三级日产三级99 | 午夜视频在线观看一区二区三区| 欧美精品a∨在线观看不卡| 成人av番号网|