简体   繁体   English

Linq Lambda表达式-查找与当前日期最接近的星期几(现在)

[英]Linq lambda expression - Find the closest day of week to current date (now)

I have for example Monday, Thursday and Sunday as draw dates. 例如,我有星期一,星期四和星期日作为抽奖日期。 What would be the way to find the closest dayOfWeek to Datetime.Now with Linq Lambda? 用Linq Lambda找到最接近Datetime.Now的dayOfWeek的方法是什么? I accomplished this with normal function, but I would like to know how to do it with Linq Lambda? 我可以使用正常功能完成此操作,但是我想知道如何使用Linq Lambda进行操作?

Example of my approach: 我的方法示例:

    public static DateTime GetSoonestDrawDate(this DateTime from, IEnumerable<LotteryDrawDate> drawDates)
    {

        int hour = 0;
        int minute = 0;

        bool todayDrawOnly = true;

        int difference = 7;
        int tempDifference = 0;

        int todayDay = (int)from.DayOfWeek;
        int drawCutOffDay = 0;

        if (todayDay == 0)
        {
            todayDay = 7;
        }

        var tempCutOffTime = new TimeSpan(23, 59, 59);

        foreach (var drawDate in drawDates)
        {
            // DayId is DayOfWeek
            drawCutOffDay = drawDate.DayId;

            if (drawCutOffDay < todayDay)
            {
                drawCutOffDay += 7;
            }

            tempDifference = drawCutOffDay - todayDay;

            var cutOffTime = new TimeSpan();
            cutOffTime = TimeSpan.Parse(drawDate.CutOffTime);

            if ((difference > tempDifference) || difference == 0)
            {
                // draw date is not today
                if (tempDifference != 0)
                {
                    difference = tempDifference;
                    hour = cutOffTime.Hours;
                    minute = cutOffTime.Minutes;
                    todayDrawOnly = false;
                }
                // same day, cutOffTime still available
                if ((tempDifference == 0 && cutOffTime > from.TimeOfDay))
                {
                    // we use tempCutOffTime to compare newest cutOffTime in case we have more cutOffTimes on the same day
                    // in that case we have to select the soonest cutOffTime of the day
                    if (cutOffTime < tempCutOffTime)
                    {
                        difference = tempDifference;
                        hour = cutOffTime.Hours;
                        minute = cutOffTime.Minutes;
                        todayDrawOnly = true;
                        tempCutOffTime = cutOffTime;
                    }
                }
                // same day selected only, but cutOffTime passed, so we add another week only in case there is only one draw date and draw date is on the same day
                else if (tempDifference == 0 && cutOffTime < from.TimeOfDay && todayDrawOnly == true)
                {
                    if (cutOffTime < tempCutOffTime)
                    {
                        difference = 7;
                        hour = cutOffTime.Hours;
                        minute = cutOffTime.Minutes;
                        todayDrawOnly = true;
                        tempCutOffTime = cutOffTime;
                    }
                }
            }
        }

        from = from.AddDays(difference);

        DateTime soonest = new DateTime(from.Year, from.Month, from.Day, hour, minute, 0);

        return soonest;
    }

Because you didn't show how LotteryDrawDate looks like, I prepared a little sample using DayOfWeek only. 因为您没有显示LotteryDrawDate样子,所以我仅使用DayOfWeek准备了一个小样本。 You have to extend that to look at time part by your own. 您必须扩展它以自己查看时间部分。

public static DateTime GetSoonestDrawDate(this DateTime from, IEnumerable<DayOfWeek> drawDates)
{
    var realDrawDates = drawDates.SelectMany(x => new[] { (int)x, (int)x + 7 }).OrderBy(x => x);
    var difference = realDrawDates.SkipWhile(x => x < (int)from.DayOfWeek).First() - (int)from.DayOfWeek;
    return from.AddDays(difference);
}

Little test code: 小测试代码:

var drawDates = new[] { DayOfWeek.Monday, DayOfWeek.Wednesday, DayOfWeek.Saturday };

for (int i = 0; i < 15; i++)
{
    var from = DateTime.Now.AddDays(i);
    Console.WriteLine("{0} - {1}", from.ToShortDateString(), GetSoonestDrawDate(from, drawDates).ToShortDateString());
}

prints ( from - next ): 打印( 从-下一个 ):

11/18/2013 - 11/18/2013
11/19/2013 - 11/20/2013
11/20/2013 - 11/20/2013
11/21/2013 - 11/23/2013
11/22/2013 - 11/23/2013
11/23/2013 - 11/23/2013
11/24/2013 - 11/25/2013
11/25/2013 - 11/25/2013
11/26/2013 - 11/27/2013
11/27/2013 - 11/27/2013
11/28/2013 - 11/30/2013
11/29/2013 - 11/30/2013
11/30/2013 - 11/30/2013
12/1/2013 - 12/2/2013
12/2/2013 - 12/2/2013

If you wanted a simple hardcoded list of draw days, you could do this: 如果您需要简单的硬编码抽奖日列表,可以执行以下操作:

DateTime GetNextDate(DateTime from)
{
    DayOfWeek target;
    switch (from.DayOfWeek)
    {
        case DayOfWeek.Friday:
        case DayOfWeek.Saturday:
        case DayOfWeek.Sunday:
            target = DayOfWeek.Sunday;
            break;
        case DayOfWeek.Monday:
            target = DayOfWeek.Monday;
            break;
        case DayOfWeek.Tuesday:
        case DayOfWeek.Wednesday:
        case DayOfWeek.Thursday:
            target = DayOfWeek.Thursday;
            break;
        default:
            throw new ArgumentException("from");
    }
    while (from.DayOfWeek != target)
        from = from.AddDays(1);
    return from;
}

A simple method to find the nearest day in the future would be the following: 查找将来的最近日期的一种简单方法是:

DayOfWeek[] draw_days = { DayOfWeek.Sunday, DayOfWeek.Monday, DayOfWeek.Thursday };
Console.WriteLine(
  "Nearest Draw Date: {0}",
  draw_days.Min(d => (d <= DateTime.Now.DayOfWeek) ? (d + 7) : d)
);

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM