22.4. Interfaces In The Wild¶
The first situations where you’ll want to use interfaces involve applying pre-defined interfaces
and classes that are part of C#. In fact, you have already encountered implementations of several these
System
provided interfaces.
22.4.1. IComparer<T>¶
Purpose: A class implements this interface to compare two objects of a given class. This is
a parameterized interface, which means that when using it, you need to specify the class that
it will be comparing. For example, IComparer<Job>
would compare Job
objects.
Important Methods and Properties: Compare(T, T)
IComparer.Compare Documentation
This interface can be used to determine, given two objects of the given type, which one is greater than the other. It is also used by collections such as List to sort its contents with the Sort method.
Using IComparer<T>
is a two step process. You must first create a class that implements the interface.
Then, provide a custom implementation of the Compare
method:
Example
public class JobComparer : IComparer<Job>
{
public int Compare(Job x, Job y)
{
// ^^ This Compare method is an implementation of IComparer.Compare
// Here, we write our own logic for comparing two job objects.
// For example, if we want to compare Job objects by name values, we'd write:
return string.Compare(x.Name, y.Name);
// ^^ Note, this Compare method is the built-in string method
// if we want to compare Jobs based on multiple fields, we can do so by expanding the custom logic
// in this ``IComparer.Compare`` implementation
}
}
Compare(T, T)
returns an integer which determines which of the two objects comes before (in other
words, “is less than”) the other. If the returned value is less than zero, then the first parameter
comes before the second. If the integer is zero, then they are considered the same. If the integer is
greater than zero, then the second parameter comes before the first.
You can think of the result of calling Compare(x, y)
as being the value of subtracting, like
x - y
. If x
is smaller than y
, this value is negative. If x
is larger than y
,
this value is positive.
22.4.2. IEnumerable<T>¶
Purpose: Enable iteration over a collection of objects using
foreach
.
Important Methods and Properties: MoveNext()
, Current
This interface is implemented by the List<T>
class, which we’ve been
using throughout this course.
Example
IEnumerable<string> collection = new List<string>();
// add items to the collection
foreach (string item in collection)
{
// do something with the item
}
Indeed, if you so desire, you may replace other instances of List
with this interface as long as those
interfaces don’t require specifically List
methods.
22.4.3. IList<T>¶
Purpose: Enable access to objects in a collection by index.
Important Methods and Properties: Add(T)
, Contains(T)
,
Remove(T)
, Count
This interface is also implemented by the List<T>
class, which we’ve
been using throughout this course. In fact, IList<T>
extends
IEnumerable<T>
. An interface may extend another interface, in the
same way that classes may extend each other.
Example
IList<string> collection = new List<string>();
// Add items to the collection
string firstItem = collection[0];
22.4.4. IDictionary<TKey, TValue>¶
Purpose: Represent a collection of key/value pairs.
Important Methods and Properties: Add(TKey, TValue)
,
Contains(T)
, Remove(T)
, Count
, Keys
, Values
IDictionary<TKey, TValue> Documentation
This interface is implemented by the Dictionary<TKey, TValue>
class,
which we’ve been using throughout this course.
Example
IDictionary<string, string> collection = new Dictionary<string, string>();
// Add items to the collection
// Get item with key "hello"
string hello = collection["hello"];
22.4.5. Check Your Understanding¶
Question
True or False
An interface can extend another interface.