Combine linq list optimisations#5777
Conversation
|
cc: @VSadov |
|
LGTM nice change! |
|
@JonHanna, I'm going to go through and review, but could rebase first to resolve the merge conflicts? |
61722b4 to
e981b71
Compare
|
Nearly managed to have the simultaneous PRs out of each others way, but for one line. |
There was a problem hiding this comment.
Should SkipListIterator really be named SkipTakeListIterator?
There was a problem hiding this comment.
Probably a better name. The current one shows its legacy more than its function.
|
Seems like there are some non-ASCII characters in the commit title: |
|
Thanks for working on this. Seems like there are several non-trivial issues to address before this can be merged. |
|
Hmm. A Left-To-Right Mark got into the title. I don't know what keypress makes that happen! |
dd9bc31 to
b6b1e49
Compare
|
@stephentoub I've included the |
Thanks, though would it not make more sense to use a TryGetCount method like I suggested and avoid the additional interface call? Are there cases where you'd want to know it's cheap and not actually get it? |
|
The problem with that approach is that there are two possibilities:
All of the changes here improve the second case, which is the only case currently used. The first would come up if we wanted to handle a subsequent operation on an Maybe what I should do is |
b6b1e49 to
4e80453
Compare
|
I've done that. It does seem to give the best of both worlds. |
|
Test Innerloop OSX Debug Build and Test |
There was a problem hiding this comment.
Does it look like this is always called with "false"?
There was a problem hiding this comment.
Currently, yes. Through the comment trail, first I had this as a Count property and @stephentoub pointed out that someone might use it as a false economy in an optimisation that would be worse when that property wasn't cheap. I said that while it wasn't necessary yet, I'd plans for that (I'm holding back on further changes that build on this as the changes will be massive if I put them all in one PR) and brought a property that reported on whether Count was cheap or not. Stephen pointed out that this would mean two interface calls would be used, and agreeing with him I changed to this. As of right now though, it's still only called with false.
|
Test Innerloop CentOS7.1 Release Build and Test |
|
Test Innerloop CentOS7.1 Release Build and Test |
|
Innerloop CentOS7.1 Release Build and Test |
|
@dotnet-bot test this please |
1 similar comment
|
@dotnet-bot test this please |
|
Test Innerloop CentOS7.1 Release Build and Test |
| ICollection<TSource> collectionoft = source as ICollection<TSource>; | ||
| if (collectionoft != null) return collectionoft.Count; | ||
| IIListProvider<TSource> listProv = source as IIListProvider<TSource>; | ||
| if (listProv != null) return listProv.GetCount(false); |
There was a problem hiding this comment.
Nit: would you mind using a named argument at the call site so that it's clear what the Boolean means? (Same for all the occurrences of this).
There was a problem hiding this comment.
Sure. Of the three places this was called that weren't passing through that value, there was this and two cases where ListPartition called it on itself. I've made those two into a call to a private Count property, to make it clearer that the flag isn't relevant to that use.
That leaves this as the sole actual use of it, but I have plans for it 😄
6871a86 to
cd1d2b4
Compare
| { | ||
| IIListProvider<TElement> listProv = source as IIListProvider<TElement>; | ||
| if (listProv != null) return listProv.GetCount(onlyIfCheap); | ||
| return !onlyIfCheap || source is ICollection || source is ICollection<TElement>? source.Count() : -1; |
There was a problem hiding this comment.
I'll also swap the order of checking ICollection<TElement> and ICollection. If Count() takes that as the sensible order to check them, I should either do the same or else demonstrate why they should both be changed.
6475a2b to
896553a
Compare
| if (partition != null) return partition.Take(count); | ||
| return TakeIterator(source, count); | ||
| IList<TSource> sourceList = source as IList<TSource>; | ||
| return sourceList != null ? new ListPartition<TSource>(sourceList, 0, count - 1) : TakeIterator(source, count); |
There was a problem hiding this comment.
Nit: FWIW, in cases like this where there's already a bunch of cases like:
IInterface1 i1 = source as IInterface1;
if (i1 != null) return i1.Whatever();
IInterface1 i2 = source as IInterface2;
if (i2 != null) return i2.Whatever();
...etc., I'd find the last one easier to understand as:
IInterfaceN iN = source as IInterfaceN;
if (iN != null) return iN.Whatever();
return Whatever();rather than:
IInterfaceN iN = source as IInterfaceN;
return iN != null ? iN.Whatever() : Whatever();Normally I like ?:, but in this case it breaks from the pattern used above it.
There was a problem hiding this comment.
I agree. Skip() ended up with a ?: from the way its changes happened, and this was analogous and so ended up with it too, but I'll change both of them to the more consistent pattern.
Anything that can serve as one can serve as the other, and also provide a faster path for Count(). Merge the two interfaces and add a Count property. Have IList optimised result of Skip() partitionable. Optimisation of Skip() for IList sources from dotnet#4551 fits with optimisations of Skip() and Take() for other sources from dotnet#2401. Combine the approaches, extending how the result of Skip() on a list handles subsequent operations.
896553a to
a087c2d
Compare
|
Test Innerloop Windows_NT Debug Build and Test please, because I'm not optimistic enough to ask about OpenSUSE. |
|
LGTM |
|
@VSadov, look good to you? |
|
LGTM |
Combine linq list optimisations
…_optimisations Combine linq list optimisations Commit migrated from dotnet/corefx@6a0c20d
A few different optimisations to Linq added recently can be usefully combined with each other.
Combine
IArrayProviderandIListProviderAnything that can be one can be the other, so merge the two interfaces into
IIListProvider.Anything implementing these can also fast-path
Count(), so do so.Have
IList<T>optimised result ofSkip()partitionable.Optimisation of
Skip()forIList<T>sources from #4551 fits with optimisations ofSkip()andTake()for other sources from #2401.Combine the approaches, extending how the result of
Skip()on a list handles subsequent operations