如何构建满足业务规则的 F# 类型?

Jak*_*and 2 c# f# domain-driven-design options optional

我正在尝试在 F# 中构建一个类型,当我获得该类型的对象时,我可以确定它处于有效状态。
该类型被调用JobId,它只包含一个Guid.
业务规则是:它必须是一个 Guid - 但不能是空的 Guid。
我已经在 C# 中实现了该类型,但现在我想将它移植到 F# 类库中。

那是 C# 类型:

public sealed class JobId
{
    public string Value { get; }

    private JobId(string value)
        => Value = value;

    public static JobId Create()
        => new JobId(Guid.NewGuid().ToString("N"));

    public static Option<JobId> Create(Guid id)
        => id == Guid.Empty
        ? None
        : Some(new JobId(id.ToString("N"));

    public static Option<JobId> Create(string id)
    {
        try
        {
            var guid = new Guid(id);
            return Create(guid);
        }
        catch (FormatException)
        {
            return None;
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

那么我如何在 F# 中构建它呢?谢谢!

更新 1:
我尝试将其实现为可区分的联合类型,如下所示:

type JobId =
    | JobId of string
Run Code Online (Sandbox Code Playgroud)

但问题是,我无法用这种方法定义任何业务规则。
所以,最后的问题是:如何确保stringJobIdIST在一定的格式?

The*_*Fox 5

例如,我已经调整了 Tomas 的答案以使用 DU 而不是类来保持适当的相等性和比较,允许JobId按预期作为分组键工作。

[<AutoOpen>]
module JobId =
    open System
    type JobId = private JobId of string with
        static member Create() = JobId(Guid.NewGuid().ToString("N"))

        static member Create(id:Guid) =
            if id = Guid.Empty then None
            else Some(JobId(id.ToString("N")))

        static member Create(id:string) =
            try JobId.Create(Guid(id))
            with :? FormatException -> None
Run Code Online (Sandbox Code Playgroud)

您必须将类型放在模块内,然后您不能直接在该模块外访问 DU 构造函数:

JobId.Create (System.Guid.NewGuid()) // Some (JobId "1715d4ae776d441da357f0efb330be43")
JobId.Create System.Guid.Empty // None
JobId System.Guid.Empty // Compile error
Run Code Online (Sandbox Code Playgroud)

  • 我通过添加扩展方法 ```[&lt;Extension&gt;] static member GetValue(jobId : JobId) : string = match jobId with | 解决了这个问题 JobId.JobId id -&gt; id``` (3认同)