不积跬步无以至千里,不积小流无以成江河。|

youliCC

园龄:5年8个月粉丝:8关注:36

规约 Specification

第一次看到这个词 Specification 是在ABP框架中,起先没怎么太关注这个。

随着接触DDD领域驱动越来越来多,关于如何集中业务逻辑的问题,再次浮现在我脑中。

好吧,我不知道。

我知道如何使用很挫的方式集中业务逻辑,问题是业务越来越大时,“这种很挫的方式集中业务逻辑” 的这一种方法将会使业务逻辑代码变得乱七八糟。

而这种乱七八糟的代码的终点是 “屎山” 。没错,“屎山” 就是巨难整,看着就让人生厌的东西,如果有位后来者看见这玩意,我相信他会骂上一句,然后......

但愿这次学习和以后的使用 Specification 可以帮助我更上一层楼吧。

ISpecification<T> 规约定义

    public interface ISpecification<T>
    {
        bool IsSatisfiedBy(T entity);

        Expression<Func<T, bool>> ToExpression();
    }

Specification<T>抽象类

复制代码
    /// <summary>
    /// This base specification implements the IsSatisfiedBy method by compiling the expression from ToExpression.
    ///
    /// This is useful for general specifications to prevent duplicated logic.
    /// Beware that it is not very performant in situations where many specifications are dynamically constructed and combined.   
    /// </summary>
    /// <remarks>
    /// see this project https://github.com/elsa-workflows/elsa-core .
    /// </remarks>
    public abstract class Specification<T> : ISpecification<T>
    {
     private Func<T, bool>? _predicate;
public virtual bool IsSatisfiedBy(T entity) { _predicate ??= ToExpression().Compile(); return _predicate(entity); } public abstract Expression<Func<T, bool>> ToExpression(); public static implicit operator Expression<Func<T, bool>>(Specification<T> specification) { return specification.ToExpression(); } }
复制代码

具体例子

例如,创建一Project类

复制代码
    public class Project
    {
        public string Id { get; set; }

        public string Name { get; set; }

        public int Size { get; set; }

        public ProjectType ProjectType { get; set; }
    }

    public enum ProjectType
    {
        A,
        B,
        C,
    }
复制代码

创建一个用于判断Project  id的规约类

复制代码
    public class ProjectIdSpecification : Specification<Project>
    {
        public string ProjectId { get; }

        public ProjectIdSpecification(string projectId)
        {
            ProjectId = projectId;
        }

        public override Expression<Func<Project, bool>> ToExpression()
        {
            return project => project.Id == ProjectId;
        }
    }
复制代码
 

数据列表

复制代码
List<Project> projects = new List<Project>()
{
    new Project()
    {
        Id = "1",
        Name ="1",
        ProjectType = ProjectType.A,
        Size = 10
    },
    new Project()
    {
        Id = "2",
        Name ="2",
        ProjectType = ProjectType.B,
        Size = 20
    },
    new Project()
    {
        Id = "3",
        Name ="3",
        ProjectType = ProjectType.C,
        Size = 30
    },
    new Project()
    {
        Id = "4",
        Name ="4",
        ProjectType = ProjectType.C,
        Size = 20
    }
};
复制代码
复制代码
// 创建一个Id的规约
var
pis = new ProjectIdSpecification("1");
// bool IsSatisfiedBy(T entity); 这个规约中的定义,其意思,传入的 T 类型是否满足此规约条件,

// 这的意思是,project[0] 的Id是否等于"1" Console.WriteLine(pis.IsSatisfiedBy(projects[
0]));
// 规约可以生成 表达式树
var cc = pis.ToExpression();
// 表达式树编译为 委托
// Expression<Func<T, bool>> ToExpression();
var aa = cc.Compile(); Console.WriteLine(aa);
// Func<T, bool> 可以加入Linq中。
var aaaa = projects.Where(aa).ToList(); Console.WriteLine(cc);
复制代码

规约的重点在于其定义的

bool IsSatisfiedBy(T entity);

Expression<Func<T, bool>> ToExpression();

一个判断传入的参数是否满足当前的规约,另一个用于生成表达式,可以在EF的表达式中使用。

不同的规约是可以两两组合使用,可以参考我的代码,其中例子更多 ,https://github.com/qiqiqiyaya/Learning-Case/tree/main/Specification

  

posted @   youliCC  阅读(87)  评论(0编辑  收藏  举报
点击右上角即可分享
微信分享提示
评论
收藏
关注
推荐
深色
回顶
收起