5

为了清楚起见,按照建议将问题总结在一行中:

我想找到对某个字段具有不同值且相隔不到一周的所有事物(基于另一个字段)


假设我有一张Users桌子和一User堂课。

每个User都有以下字段:

  • SignUpDate非空DateTime字段
  • UserType一个 int 字段,对于这个问题,假设它是1or0

我想选择所有注册时间少于 7 天且具有不同用户类型的用户。

我当前的(悲惨的)尝试包括一个OrderBy(r=>r.SignUpDate)和一个 .ToList,这并不是什么大问题,因为每两周的用户数量并不大。(我从重叠的几周中抓取了所有用户并比较了数据)。

但是,我发现我目前的解决方案很差。我不知道解决这个问题的正确方法是什么。

我认为这里的核心问题是我不知道如何在订购后解决 LINQ to Entities 中“选择每两个对应记录”的概念。

非常感谢任何帮助,我对如何在将来不开始自己的问题的情况下解决此类问题感兴趣。

示例输入

 SignUpDate        UserType
 ------------------------------------
 2008-11-11          1
 2008-11-12          0
 2008-11-13          0
 2008-12-13          0
 2008-12-15          1

示例输出

任何有意义的方式来表明有问题的对是:

 2008-11-11          1
 2008-11-12          0

(不同的日子和不同的类型)

 2008-12-13          0
 2008-12-15          1

(不同的两天和不同的类型)


这是我找到的相关 SQL 解决方案。

4

2 回答 2

3

我不完全理解您要解决的问题,因此以下只是一般性建议。听起来任何两个“时间相邻”并且彼此在一周内的用户注册都是规则,但这听起来有点奇怪..

每当您想查询仅间接可用的信息(即不是简单的列值的信息)时,请使用投影来选择解决问题所需的信息。

var query = from user in context.Users
            let previousUser = context.Users
                .Where( u => u.SignUpDate < user.SignUpDate )
                .OrderBy( u => u.SignUpDate )
                .FirstOrDefault()
            select new
            {
                User = user,
                PreviousUser = previousUser,
                IsDuplicate = previousUser != null && previousUser.UserType != user.UserType,
                SignUpDaysApart = user.SignUpDate.Subtract( previousUser.SignUpDate )
            };

一旦您拥有更易于访问的格式的数据,编写查询来解决业务问题就会变得更加容易。

var duplicates = (from d in query
                 where d.IsDuplicate && d.SignUpDaysApart.TotalDays <= 7
                 select d).ToList();

我希望以上内容足以启发您找到解决方案。

于 2013-06-09T23:32:07.500 回答
0

想出了这样的东西

        private bool LessThan7DaysApart(DateTime d1, DateTime d2)
        {
            return (d1 - d2).Duration() < new TimeSpan(7, 0, 0, 0);
        }


        private void Match()
        {
            List<User> listfortype0 = users.Where(u => u.UserType == 0).ToList();
            List<User> listfortype1 = users.Where(u => u.UserType == 1).ToList();

            foreach (User u in listfortype0)
            {
                List<User> tmp = listfortype1.Where(u2 => LessThan7DaysApart(u2.SignUpDate, u2.SignUpDate)).ToList();
                if (tmp.Count > 0)
                {
                    List<User> matchedusers = new List<User> { u, tmp[0] };                    
                    listfortype1.Remove(tmp[0]);
                }
            }
        }
于 2013-06-09T23:24:40.393 回答