namespace System.Immutable {
public interface IImmutableEquatable<T> : IEquatable<T> { };
public static partial class ExtensionMethods {
public static bool Equals<T>(this T inst, object obj, Func<T, bool> thisEquals) where T : IImmutableEquatable<T> =>
object.ReferenceEquals(inst, obj)
&& obj.GetType() == inst.GetType()
&& inst.GetHashCode() == obj.GetHashCode()
public static int GetHashCode<T>(this T inst, ref int? hashCache, Func<int> thisHashCode) where T : IImmutableEquatable<T> {
if (hashCache is null) hashCache = thisHashCode();
class Base: IImmutableEquatable<Base> {
public Base(int X, int Y) {
public bool ThisEquals(Base o) {
Console.WriteLine(" Base _Equals");
return X.Equals(o.X) && Y.Equals(o.Y);
public int ThisHashCode() {
Console.WriteLine(" Base _GetHashCode");
return Tuple.Create(X, Y).GetHashCode();
public override bool Equals(object obj) => this.Equals(obj, ThisEquals);
public bool Equals(Base o) { Console.WriteLine(" Base.Equals(Base)"); return object.Equals(this, o); }
public static bool operator ==(Base o1, Base o2) { Console.WriteLine(" Base=="); return object.Equals(o1, o2); }
public static bool operator !=(Base o1, Base o2) { Console.WriteLine(" Base!="); return !object.Equals(o1, o2); }
protected int? hashCache;
public override int GetHashCode() => this.GetHashCode(ref hashCache, ThisHashCode);
class Derived : Base, IImmutableEquatable<Derived> {
public Derived(int X, int Y, int Z, int K) : base(X, Y) {
public bool ThisEquals(Derived o) {
Console.WriteLine(" Derived _Equals");
return base.ThisEquals(o) && Z.Equals(o.Z) && K.Equals(o.K);
public new int ThisHashCode() {
Console.WriteLine(" Derived _GetHashCode");
return Tuple.Create(base.ThisHashCode(), Z, K).GetHashCode();
public override bool Equals(object obj) => this.Equals(obj, ThisEquals);
public bool Equals(Derived o) { Console.WriteLine(" Derived.Equals(Derived)"); return object.Equals(this, o); }
public override int GetHashCode() => this.GetHashCode(ref hashCache, ThisHashCode);
public static void Main()
Derived d1 = new Derived(0, 1, 2, 3), d2 = new Derived(0, 1, 4, 5), d3 = new Derived(0, 1, 2, 3), d4 = null;
Base b1 = d1, b2 = d2, b3 = d3, b4 = d4, b5=null;
Console.WriteLine("d1==d2"); Console.WriteLine(" "+ (d1==d2));
Console.Write("d1.Equals(d2)"); Console.WriteLine(" " + d1.Equals(d2)+"\n");
Console.WriteLine("d1==d3"); Console.WriteLine(" "+ (d1==d3));
Console.Write("d1.Equals(d3)"); Console.WriteLine(" " + d1.Equals(d3)+"\n");
Console.WriteLine("d1==d4"); Console.WriteLine(" "+ (d1==d4));
Console.Write("d1.Equals(d4)"); Console.WriteLine(" " + d1.Equals(d4)+"\n");
Console.WriteLine("b1==b2"); Console.WriteLine(" "+ (b1==b2));
Console.Write("b1.Equals(b2)"); Console.WriteLine(" " + b1.Equals(b2)+"\n");
Console.WriteLine("b1==b3"); Console.WriteLine(" "+ (b1==b3));
Console.Write("b1.Equals(b3)"); Console.WriteLine(" " + b1.Equals(b3)+"\n");
Console.WriteLine("b1==b4"); Console.WriteLine(" "+ (b1==b4));
Console.Write("b1.Equals(b4)"); Console.WriteLine(" " + b1.Equals(b4)+"\n");
Console.WriteLine("b==d1"); Console.WriteLine(" "+ (b==b1));
Console.Write("b.Equals(d1)"); Console.WriteLine(" " + b.Equals(d1)+"\n");
Console.WriteLine("b5==d1"); Console.WriteLine(" "+ (b5==d1));
Console.WriteLine("b5==d4"); Console.WriteLine(" "+ (b5==d4));