Can I use LINQ to retrieve only “on change” values?

前端 未结 5 828
死守一世寂寞
死守一世寂寞 2021-01-22 20:36

What I\'d like to be able to do is construct a LINQ query that retrieved me a few values from some DataRows when one of the fields changes. Here\'s a contrived example to illus

相关标签:
5条回答
  • 2021-01-22 21:17

    You could use the IEnumerable extension that takes an index.

    var all = ds.Tables[0].AsEnumerable();
    var weatherStuff = all.Where( (w,i) => i == 0 || w.Field<string>("Observation") != all.ElementAt(i-1).Field<string>("Observation") );
    
    0 讨论(0)
  • 2021-01-22 21:19

    Here is one more general thought that may be intereting. It's more complicated than what @tvanfosson posted, but in a way, it's more elegant I think :-). The operation you want to do is to group your observations using the first field, but you want to start a new group each time the value changes. Then you want to select the first element of each group.

    This sounds almost like LINQ's group by but it is a bit different, so you can't really use standard group by. However, you can write your own version (that's the wonder of LINQ!). You can either write your own extension method (e.g. GroupByMoving) or you can write extension method that changes the type from IEnumerable to some your interface and then define GroupBy for this interface. The resulting query will look like this:

    var weatherStuff = 
      from row in ds.Tables[0].AsEnumerable().AsMoving()
      group row by row.Field<string>("Observation") into g
      select g.First();
    

    The only thing that remains is to define AsMoving and implement GroupBy. This is a bit of work, but it is quite generally useful thing and it can be used to solve other problems too, so it may be worth doing it :-). The summary of my post is that the great thing about LINQ is that you can customize how the operators behave to get quite elegant code.

    I haven't tested it, but the implementation should look like this:

    // Interface & simple implementation so that we can change GroupBy
    interface IMoving<T> : IEnumerable<T> { }
    class WrappedMoving<T> : IMoving<T> {
      public IEnumerable<T> Wrapped { get; set; }
      public IEnumerator<T> GetEnumerator() { 
        return Wrapped.GetEnumerator(); 
      }
      public IEnumerator<T> GetEnumerator() { 
        return ((IEnumerable)Wrapped).GetEnumerator(); 
      }
    }
    
    // Important bits:
    static class MovingExtensions { 
      public static IMoving<T> AsMoving<T>(this IEnumerable<T> e) {
        return new WrappedMoving<T> { Wrapped = e };
      }
    
      // This is (an ugly & imperative) implementation of the 
      // group by as described earlier (you can probably implement it
      // more nicely using other LINQ methods)
      public static IEnumerable<IEnumerable<T>> GroupBy<T, K>(this IEnumerable<T> source, 
           Func<T, K> keySelector) {
        List<T> elementsSoFar = new List<T>();
        IEnumerator<T> en = source.GetEnumerator();
        if (en.MoveNext()) {
          K lastKey = keySelector(en.Current);
          do { 
            K newKey = keySelector(en.Current);
            if (newKey != lastKey) { 
              yield return elementsSoFar;
              elementsSoFar = new List<T>();
            }
            elementsSoFar.Add(en.Current);
          } while (en.MoveNext());
          yield return elementsSoFar;
        }
      }
    
    0 讨论(0)
  • 2021-01-22 21:22

    This is one of those instances where the iterative solution is actually better than the set-based solution in terms of both readability and performance. All you really want Linq to do is filter and pre-sort the list if necessary to prepare it for the loop.

    It is possible to write a query in SQL Server (or various other databases) using windowing functions (ROW_NUMBER), if that's where your data is coming from, but very difficult to do in pure Linq without making a much bigger mess.


    If you're just trying to clean the code up, an extension method might help:

    public static IEnumerable<T> Changed(this IEnumerable<T> items,
        Func<T, T, bool> equalityFunc)
    {
        if (equalityFunc == null)
        {
            throw new ArgumentNullException("equalityFunc");
        }
        T last = default(T);
        bool first = true;
        foreach (T current in items)
        {
            if (first || !equalityFunc(current, last))
            {
                yield return current;
            }
            last = current;
            first = false;
        }
    }
    

    Then you can call this with:

    var changed = rows.Changed((r1, r2) =>
        r1.Field<string>("Observation") == r2.Field<string>("Observation"));
    
    0 讨论(0)
  • 2021-01-22 21:23

    I think what you are trying to accomplish is not possible using the "syntax suggar". However it could be possible using the extension method Select that pass the index of the item you are evaluating. So you could use the index to compare the current item with the previous one (index -1).

    0 讨论(0)
  • 2021-01-22 21:24

    You could useMorelinq's GroupAdjacent() extension method

    GroupAdjacent: Groups the adjacent elements of a sequence according to a specified key selector function...This method has 4 overloads.

    You would use it like this with the result selector overload to lose the IGrouping key:-

    var weatherStuff = ds.Tables[0].AsEnumerable().GroupAdjacent(w => w.Field<string>("Observation"), (_, val) => val.Select(v => v));
    

    This is a very popular extension to default Linq methods, with more than 1M downloads on Nuget (compared to MS's own Ix.net with ~40k downloads at time of writing)

    0 讨论(0)
提交回复
热议问题