Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How to pass a List<Child> to a method with parameter List<Parent>?

I've been away from inheritance for a while and need a little helping hand.

I have an abstract base class Chief. There are two inheriting classes Ship and Vehicle, which share several properties via Chief.

I have a method which uses those common properties:

public static void AssignRandomProperties(PSetList routeSettings, List<Chief> theChiefs)

but when I try to pass a

List<Ship> 

or a

List<Vehicle> 

I'm told that they can't be converted. Here's a sample method call:

ChiefRouteRandomizer.AssignRandomProperties(s.Route, theVehicles);

I was assuming of course that the List<Vehicle> would be treated as a List<Chief> when passed as the argument to AssignRandomProperties, but apparently not. Does the list need converting to List<Chief> first? If so, how?

like image 815
MrGreggles Avatar asked Dec 19 '10 06:12

MrGreggles


3 Answers

First some explanation. You cannot convert List<Vehicle> to List<Chief> because then the Add method would have the signature void List<Chief>.Add(Chief item) and you would be able to store instances of the Chief class in a list that was declared to only hold Vehicles. This would break the type system; therefore it is not allowed.

The easiest way to get around this is to pass in theVehicles.ToList<Chief>(). However, this will create a new list. This has two implications: (1) You would be wasting memory by duplicating the list, and (2) if the method is going to mutate the list itself (add/remove items or replace members with other members) you will not see those changes on the theVehicles list. (If the objects the references in the list point to are the only things being modified, this is not a problem.)

If you have control over the AssignRandomProperties method, consider using this signature instead:

public static void AssignRandomProperties<T>(
    PSetList routeSettings,
    IList<T> theChiefs) where T : Chief
{
    ...
}
like image 199
cdhowie Avatar answered Nov 05 '22 23:11

cdhowie


Look at this:

List<Ship> ships = new List<Ship>();
List<Chief> asChiefs = (List<Chief>)ships;
asChiefs.Add(new Car());
// so what, now ships[0] is a Car?

That's why you cannot treat a list of ships as a list of chiefs. If you want to learn more about this, it is said that List is not covariant (and neither is it contravariant).

Now, does your AssignRandomProperties method really need to take a List<T>? If all you do is iterate it, you don't need a List<T>, you can do fine with an IEnumerable<T>, which List<T> implements. The good thing here is IEnumerable<T> provides no methods for modification of the list, so it is covariant (starting with .NET 4). This means you can do this:

IEnumerable<Chief> chiefs = ships;
like image 29
R. Martinho Fernandes Avatar answered Nov 05 '22 23:11

R. Martinho Fernandes


Here is my favorite trick

public static void AssignRandomProperties<T>(PSetList routeSettings, List<T> theChiefs) 
    where T : Chief

Then, you call like following

AssignRandomProperties<Vehicle>(s.Route, theVehicles);
like image 1
Harvey Kwok Avatar answered Nov 06 '22 01:11

Harvey Kwok