ASP.NET Core模仿中間件方式實現列表過濾功能

我們的很多功能當中都會遇到對版本進行過濾的場合,例如你可能需要對列表中的數據的時間進行過濾、版本過濾、渠道以及地區信息進行過濾。

原本的做法:設計很多個過濾方法,通過枚舉的方式組合,選擇需要過濾哪些方法,然後一個方法一個方法的調用。 這樣的做法本身沒什麼問題。但是感覺很面向過程,不夠面向對象。

通過學習.Net Core的源碼,我們可以瞭解到它采用瞭一種委托鏈表的方式,將所有的中間件都串瞭起來。所以我想要仿造它這個去實現一下這個功能。

這樣做的好處:抽象出一些過濾的方法,對於不同的系統,隻要通過Use方法,就可以增加我們的過濾規則,考慮到不同的系統過濾的規則不同,這樣做也比較靈活(例如 應用管理系統 需要過濾版本、渠道、地區 ,而黑白名單需要過濾版本、渠道、時間等等,那麼對於前者我隻需要在過濾的時候 UseVersion UserChannel UseArea, 對於後者把UserArea緩存UseTime即可。)

廢話不多說:上碼

1.定義一個委托類型,承載我們過濾方法

namespace FilterDelegate
{
      public delegate IEnumerable<TcySysApplication> TcySysFilterDelegate(IEnumerable<TcySysApplication> applist, TcySysFilterConditionInfo conditionInfo );
}

委托的輸入是我們待處理的數據列表appList, 以及我們執行過濾的條件數據。

2.定義一個Builder類,主要用於構建我們整個過濾器,裡面主要有兩個方法Use方法以及Build方法,Use方法主要用於往我們的委托列表裡面增加過濾委托,Build方法主要用於生成最後的過濾器

private readonly IList<Func<TcySysFilterDelegate, TcySysFilterDelegate>> _components = new List<Func<TcySysFilterDelegate, TcySysFilterDelegate>>();
 
public TcySysFilterBuilder()
{
 
}
 
public TcySysFilterBuilder Use(Func<TcySysFilterDelegate, TcySysFilterDelegate> filterItem)
{
    _components.Add(filterItem);
    return this;
}
 
public TcySysFilterDelegate Build()
{
    TcySysFilterDelegate last = (applist,filterInfo) =>
    {
        Console.WriteLine("過濾完成");
        return applist;
    };
 
    foreach (var component in _components.Reverse())
    {
        last = component(last);
    }
 
    return last;
}

3.定義一堆過濾方法,這裡沒有寫具體的邏輯,每一個方法都是通過Builder.Use將委托加入到委托鏈中

public static TcySysFilterBuilder UseTimeFilter(this TcySysFilterBuilder builder)
{
    return builder.Use(next =>
    {
        return (list, filterInfo) =>
        {
            Console.WriteLine("我是時間過濾");
            return next(list, filterInfo);
        };
    });
}
 
 
public static TcySysFilterBuilder UseChannelFilter(this TcySysFilterBuilder builder)
{
    return builder.Use(next =>
    {
        return (list, filterInfo) =>
        {
            Console.WriteLine("我是渠道過濾");
 
            list = list.Where(x => x.ChannelId != filterInfo.ChannelId);
 
            return next(list, filterInfo);
        };
    });
}
 
public static TcySysFilterBuilder UseVersionFilter(this TcySysFilterBuilder builder)
{
    return builder.Use(next =>
    {
        return (list, filterInfo) =>
        {
            Console.WriteLine("我是版本過濾");
            return next(list, filterInfo);
        };
    });
}

4.其他類型

public class TcySysApplication
{
    public long AppId { set; get; }
    public string Name { set; get; }
    public long ChannelId { set; get; }
    public long Version { set; get; }
    public string Province { set; get; }
    public string City { set; get; }
    public string District { set; get; }
}
 
public class TcySysFilterConditionInfo
{
    public long ChannelId { set; get; }
    public long Version { set; get; }
    public string Province { set; get; }
    public string City { set; get; }
    public string District { set; get; }
}

5.執行使用

class Program
{
    static void Main(string[] args)
    {
        var sourceList = new List<TcySysApplication>();
 
        sourceList.Add(new TcySysApplication {
            AppId =1000,
            ChannelId = 88215,
            District = "",
            City = "烏魯木齊",
            Province = "新疆",
            Name ="愛玩不玩",
            Version = 10001
        });
 
        sourceList.Add(new TcySysApplication
        {
            AppId = 1001,
            ChannelId = 310200,
            District = "",
            City = "烏魯木齊",
            Province = "新疆",
            Name = "愛玩不玩2",
            Version = 10002
        });
 
 
        var filterInfo = new TcySysFilterConditionInfo
        {
            ChannelId = 310200,
            District = "",
            City = "北京",
            Province = "北京",
            Version = 10002
        };
 
 
        var builder = new TcySysFilterBuilder();
 
        builder.UseTimeFilter()
            .UseChannelFilter()
            .UseVersionFilter();
 
        var filter = builder.Build();
 
        var result = filter(sourceList, filterInfo);
 
 
        foreach (var item in result)
        {
            Console.WriteLine($"AppId={item.AppId} AppName={item.Name} ChannelId={item.ChannelId}");
        }
 
        Console.ReadKey();
    }
}

圖上我使用瞭三種過濾,其中因為Channel中有過濾的邏輯,根據這個邏輯我們應該隻會返回一個Channeld = 88215的數據

我們可以將 UseChannelFilter 那句代碼註釋掉,再運行,由於沒有過濾渠道,此時顯示瞭兩條數據

以上就是這篇文章的全部內容瞭,希望本文的內容對大傢的學習或者工作具有一定的參考學習價值,謝謝大傢對WalkonNet的支持。如果你想瞭解更多相關內容請查看下面相關鏈接

推薦閱讀: