Make mutable generic collection interfaces implement read-only collection interfaces #31001
Description
Rationale
It's long been a source of confusion that the mutable generic collection interfaces don't implement their respective read-only collection interfaces. This was of course due to the read-only collection interfaces being added after the fact and thus would cause breaking changes by changing a published interface API.
With the addition of default interface implementations in C#8/.NET Core 3.0 I think the mutable generic collection interfaces, ICollection<T>
, IList<T>
, IDictionary<K, V>
, and ISet<T>
should now implicitly inherit their respective read-only collection interfaces. This can now be done without causing breaking changes.
While it would have been nice for these interfaces to share members, I think the proposed API below is the best we can possibly do with the read-only interfaces being added after the fact.
As an added bonus, this should allow some simplification of the type checking in LINQ code to check for the read-only interfaces instead of the mutable interfaces.
Proposed API
namespace System.Collections.Generic {
- public interface ICollection<T> : IEnumerable<T> {
+ public interface ICollection<T> : IReadOnlyCollection<T> {
- int Count { get; }
+ new int Count { get; }
+ int IReadOnlyCollection<T>.Count => Count;
}
- public interface IList<T> : ICollection<T> {
+ public interface IList<T> : ICollection<T>, IReadOnlyList<T> {
- T this[int index] { get; set; }
+ new T this[int index] { get; set; }
+ T IReadOnlyList<T>.this[int index] => this[index];
}
- public interface IDictionary<TKey, TValue> : ICollection<KeyValuePair<TKey, TValue>> {
+ public interface IDictionary<TKey, TValue> : ICollection<KeyValuePair<TKey, TValue>>, IReadOnlyDictionary<TKey, TValue> {
- TValue this[TKey key] { get; set; }
+ new TValue this[TKey key] { get; set; }
- ICollection<TKey> Keys { get; }
+ new ICollection<TKey> Keys { get; }
- ICollection<TValue> Values { get; }
+ new ICollection<TValue> Values { get; }
- bool ContainsKey(TKey key);
+ new bool ContainsKey(TKey key);
- bool TryGetValue(TKey key, out TValue value);
+ new bool TryGetValue(TKey key, out TValue value);
+ TValue IReadOnlyDictionary<TKey, TValue>.this[TKey key] => this[key];
+ IEnumerable<TKey> IReadOnlyDictionary<TKey, TValue>.Keys => Keys;
+ IEnumerable<TValue> IReadOnlyDictionary<TKey, TValue>.Values => Values;
+ bool IReadOnlyDictionary<TKey, TValue>.ContainsKey(TKey key) => ContainsKey(key);
+ bool IReadOnlyDictionary<TKey, TValue>.TryGetValue(TKey key, out TValue value) => TryGetValue(key, out value);
}
- public interface ISet<T> : ICollection<T> {
+ public interface ISet<T> : ICollection<T>, IReadOnlySet<T> {
- bool IsProperSubsetOf(IEnumerable<T> other);
+ new bool IsProperSubsetOf(IEnumerable<T> other);
- bool IsProperSupersetOf(IEnumerable<T> other);
+ new bool IsProperSupersetOf(IEnumerable<T> other);
- bool IsSubsetOf(IEnumerable<T> other);
+ new bool IsSubsetOf(IEnumerable<T> other);
- bool IsSupersetOf(IEnumerable<T> other);
+ new bool IsSupersetOf(IEnumerable<T> other);
- bool Overlaps(IEnumerable<T> other);
+ new bool Overlaps(IEnumerable<T> other);
- bool SetEquals(IEnumerable<T> other);
+ new bool SetEquals(IEnumerable<T> other);
// Adding this new member is required so that there's a most specific Contains method on ISet<T> since ICollection<T> and IReadOnlySet<T> define it too
+ new bool Contains(T value) => ((ICollection<T>)this).Contains(value);
+ bool IReadOnlySet<T>.Contains(T value) => ((ICollection<T>)this).Contains(value);
+ bool IReadOnlySet<T>.IsProperSubsetOf(IEnumerable<T> other) => IsProperSubsetOf(other);
+ bool IReadOnlySet<T>.IsProperSupersetOf(IEnumerable<T> other) => IsProperSupersetOf(other);
+ bool IReadOnlySet<T>.IsSubsetOf(IEnumerable<T> other) => IsSubsetOf(other);
+ bool IReadOnlySet<T>.IsSupersetOf(IEnumerable<T> other) => IsSupersetOf(other);
+ bool IReadOnlySet<T>.Overlaps(IEnumerable<T> other) => Overlaps(other);
+ bool IReadOnlySet<T>.SetEquals(IEnumerable<T> other) => SetEquals(other);
+ }
}
Binary Compatibility Test
I was able to test that this change doesn't break existing implementers with the following custom interfaces and by simply dropping the new interfaces dll to the publish folder without recompiling the consuming code, the IMyReadOnlyList<T>
interface was automatically supported without breaking the code.
Original Interfaces DLL code
namespace InterfaceTest
{
public interface IMyReadOnlyList<T>
{
int Count { get; }
T this[int index] { get; }
}
public interface IMyList<T>
{
int Count { get; }
T this[int index] { get; set; }
}
}
New Interfaces DLL code
namespace InterfaceTest
{
public interface IMyReadOnlyList<T>
{
int Count { get; }
T this[int index] { get; }
}
public interface IMyList<T> : IMyReadOnlyList<T>
{
new int Count { get; }
new T this[int index] { get; set; }
int IMyReadOnlyList<T>.Count => Count;
T IMyReadOnlyList<T>.this[int index] => this[index];
}
}
Consuming Code
using System;
using System.Collections.Generic;
namespace InterfaceTest
{
class Program
{
static void Main()
{
var myList = new MyList<int>();
Console.WriteLine($"MyList<int>.Count: {myList.Count}");
Console.WriteLine($"IMyList<int>.Count: {((IMyList<int>)myList).Count}");
Console.WriteLine($"IMyReadOnlyList<int>.Count: {(myList as IMyReadOnlyList<int>)?.Count}");
Console.WriteLine($"MyList<int>[1]: {myList[1]}");
Console.WriteLine($"IMyList<int>[1]: {((IMyList<int>)myList)[1]}");
Console.WriteLine($"IMyReadOnlyList<int>[1]: {(myList as IMyReadOnlyList<int>)?[1]}");
}
}
public class MyList<T> : IMyList<T>
{
private readonly List<T> _list = new List<T> { default, default };
public T this[int index] { get => _list[index]; set => _list[index] = value; }
public int Count => _list.Count;
}
}
Original Output
MyList<int>.Count: 2
IMyList<int>.Count: 2
IMyReadOnlyList<int>.Count:
MyList<int>[1]: 0
IMyList<int>[1]: 0
IMyReadOnlyList<int>[1]:
New Output
MyList<int>.Count: 2
IMyList<int>.Count: 2
IMyReadOnlyList<int>.Count: 2
MyList<int>[1]: 0
IMyList<int>[1]: 0
IMyReadOnlyList<int>[1]: 0
Moved from #16151
Updates
- Added
ISet<T>
implementingIReadOnlySet<T>
since Please add interface IReadOnlySet and make HashSet, SortedSet implement it #2293 has been implemented.
Activity