I've been trying to make a generic Reverse Priority Queue, but int the EnQueue, but I still can't manage the errors that come with using IComparer.
The Error: Error 1 An object reference is required for the non-static field, method, or property 'System.Collections.Generic.IComparer.Compare(T, T)'
public void inQ(T dat)//adding element in place, increasing order
{
if (start == null)//that means that the RPQ is empty
{
start = new node(dat);
return;
}
if (IComparer<T>.Compare(start.data, dat) > 0)//Doesn't work
{
start = new node(dat, start);
return;
}
//Default Case
//no need for an else, actually
node q = start;
while (q.next != null && Comparer<T>.Default.Compare(q.next.data, dat) < 0)//Works Perfectly
q++;
q.next = new node(dat, q.next);
}
What happened
you need an instance of
IComparer<T>
to be able to call the methodCompare
on it... remember this is an interface, it doesn't have static methods.That's why this doesn't work:
On the other hand
Comparer<T>
is a class and it offers you an static propertyDefault
that gives you an instance ofIComparer<T>
...That's why this works:
Getting an instance of
IComparer<T>
You can consider storing a field with your instance of
IComparer<T>
and using that instead. That will minimize confusion, and also allow you to get anIComparer<T>
in the constructor to be stored on that field - which is useful in case the client wants a custom behaviour.The default instance
If you want to use the default comparer you can get it from it:
This comparer will give the default behaviour for whatever
T
you use. This behaviour is equivalent to callingCompareTo
on the instances ofT
, with extra handling for nulls. Since you cannot use the methodCompareTo
when your intance is null you may want to check for null before callingCompareTo
... using a comparer removes the problem.Implementing
IComparer<T>
Since
IComparer<T>
is an interface, you can write a class to implement it, that class must have aCompare
method with whatever logic you want. Then you create an instance of that class and use it where the system wants anIComparer<T>
.See the an example of implementing
IComparer<T>
on MSDNYour comparer would be an instance of the class:
Note: the documentation actually suggest to inherit from
Comparer<T>
instead of implementingIComparer<T>
directly, the pragmatic reason is thatComparer<T>
also implementsIComparer
in addition ofIComparer<T>
.Creating an
IComparer<T>
with a delegateIn case you are not familiar with what a delegate is, let's say it is a reference to a method, so you can have a variable that has a reference to method and pass it around.
You can create an
IComparer<T>
if you have a delegate to a method that does the comparison... this is done by calling the methodComparer<T>.Create
witch will take a delegate to the method you want. For example you can use a lambda expresion as follows:The code above is a shorthand for this:
Which in turn is a shorthand for this:
Which is sugar for something like this (except that in the above code the method is anonymous):
Extra readings: