> 在 c# LanguageExt 中,c#,functional-programming,optional,either,language-ext"/>

首页 > 解决方案 > 转换任何一个> 到选项> 在 c# LanguageExt 中

问题描述

我正在使用 C# LanguageExt https://github.com/louthy/language-ext

我有一个MyDto从一些 Json 解析的类。解析函数返回Either<Error, Mydto>. 如果 dto 匹配给定的规则,或者它是一个错误,那么我想取回结果,否则,什么都没有。

最终结果应该是 type Option<Either<Error, Mydto>>

我最终得到了这样的东西

Option<Either<Error, MyDto>> finalResult =  
     MyDto.From("some json") // Returns Either<Error, MyDto>
          .Right(dto => (dto.Equals("something")
             ? Some<Either<Error, IDhResponse>>(Right(dto))
             : None))
          .Left(error => Some<Either<Error, IDhResponse>>(Left(error)));

我真的不喜欢它,因为它呈现出太多的重复。

然后我尝试了这个

MyDto.From("some json") // Returns Either<Error, MyDto>
   .Map(dto => dto.Equals("something")
      ? Some(dto)
      : None)

Either<Error, Option<MyDto>>如果此时我能够将 Option 带出,它返回的结果看起来还不错。不幸的是,我找不到任何可以做到这一点的东西。

有什么东西存在吗?还是存在一些更好的方法来完成我的目的?

也许像

 MyDto.From("some json") // Returns Either<Error, MyDto>
   .Where(dto => dto.Equals("something")) // Only applies to Right branch and wrap everything in  Option when fails

感谢您对此的任何建议。

标签: c#functional-programmingoptionaleitherlanguage-ext

解决方案


好的,下面是三个代码解决方案:

Either<Exception, int> resultOfParsing = Right(10); // Left(new Exception("error"))

// version 1: make use of "bottom" state:
Either<Exception, int> filterResultUsingBottom = resultOfParsing.Filter(i => i > 5);

var textResultUsingBottom = filterResultUsingBottom.Match(i => $"success: {i}", exception => $"error: {exception.Message}", () => "condition: false");

// version 2: inner option
Either<Exception, Option<int>> filterResultInnerOption = resultOfParsing
    .Map(i => Some(i))
    .FilterT(i => i > 5);

var textResultInnerOption = filterResultInnerOption.Match(noError =>
        noError.Match(i => $"success: {i}", () => "condition: false"),
    exception => $"error: {exception.Message}");

// version 3: outer option
Option<Either<Exception, int>> filterResultOuterOption = filterResultInnerOption.Sequence();

var textResultOuterOption = filterResultOuterOption.Match(noConditionFail =>
        noConditionFail.Match(i => $"success: {i}", exception => $"error: {exception.Message}"),
    () => "condition: false");

如您所见,所有版本都可以以类似的方式使用(带有 的行Match)。您应该根据您希望在程序中拥有/传递的类型来选择您需要的版本。

如果您不太关心“条件失败”和“异常”之间的区别,那么版本 1(底部)可能没问题(避免类型嵌套)。但我个人更喜欢后一种解决方案。您可以根据需要在它们之间切换.Sequence()


推荐阅读