Groups | Blog | Home
all groups > dotnet clr > november 2006 >

dotnet clr : Difficulty Extending from Generic Base Class


jkitagr
11/13/2006 7:20:01 AM
I am trying to create a library that extends from a common base class that
provides common functionality, but I am running into one significant road
block. Let me give you an example.

If I do not do this with Generics, I would do something like this (Foo and
Bar extend from BaseClass):

public abstract class BaseClass
{
}

public class Foo : BaseClass
{
public override string ToString()
{
return "I am a Foo";
}
}

public class Bar : BaseClass
{
public override string ToString()
{
return "I am a Bar";
}
}

Then I could do some things like this:

Foo fooItem = new Foo();
Console.WriteLine(fooItem.ToString());

Bar barItem = new Bar();
Console.WriteLine(barItem.ToString());

And then add them to a common generic list like this:

List<BaseClass> listOfItems = new List<BaseClass>();
listOfItems.Add(fooItem);
listOfItems.Add(barItem);

foreach (BaseClass baseItem in listOfItems)
{
Console.WriteLine(baseItem.ToString());
}

The list of abstract base class items is a concrete class. But now I want
to make the base class generic, to have access to the type information like
this:

public abstract class BaseClass<T> where T : BaseClass<T>
{
public override string ToString()
{
return "I am a " + typeof(T).Name;
}
}

public class Foo : BaseClass<Foo>
{
}

public class Bar : BaseClass<Bar>
{
}

I can still do this, and it works:

Foo fooItem = new Foo();
Console.WriteLine(fooItem.ToString());

Bar barItem = new Bar();
Console.WriteLine(barItem.ToString());

But when I try to define a list of abstract base class items like:

List<BaseClass<>> listOfItems = new List<BaseClass<>>();

It expects a type parameter to be supplied. I can make it a list of
BaseClass<Foo> or BaseClass<Bar> but I want the list to be able to contain
either type. I could make the list of object, but then I don't have the
ability to restrict the type to Foo or Bar. I think what I am looking for is
a contraint that I can place on the type supplied to the list but there does
not seem to be any syntax for this.

Am I missing something in working with Generics?

Jim

Ciaran O''Donnell
11/13/2006 7:37:02 AM
Your not missing anything apart from the diffculty in building generics to
work the way you want. BaseClass<Foo> does not inherit from BaseClass<T>,
its a different type.

Ciaran

[quoted text, click to view]
jkitagr
11/13/2006 7:55:01 AM
But when I call Foo.ToString() it is clearly calling BaseClass<T>.ToString()
so it is extending from the generic base class. Is that not what you mean?

Jim

[quoted text, click to view]
AJ
11/13/2006 4:06:25 PM
In article <30164187-B478-42A8-A41E-8BD3BB8A66DE@microsoft.com>,
jkitagr@discussions.microsoft.com says...
[quoted text, click to view]


You could have an abstract, non-generic base class from which BaseClass
<T> derives

i.e. class BaseClass<T> : BaseClass where T : BaseClass<T> {..}

and then have List<BaseClass> throughout.

Or, in a similar way, define an interface which BaseClass<T> implements,
and have a List<IBaseInterface>

I don't believe there's another way to achieve this, though. (Having hit
jkitagr
11/14/2006 1:00:01 PM
Ciaran and AJ,

Thanks for your input. It has helped in my understanding of Generics and
AddThis Social Bookmark Button