Java: Subclassing a genericised class
Asked Answered
E

2

9

I have a genericised class that I wish to subclass as follows:

public class SomeTable<T extends BaseTableEntry>
    extends BaseTable<T>
{

    public SomeTable(int rows, int cols)
    {
        super(rows, cols, SomeTableEntry.class);
        //Does not compile:
        //Cannot find symbol: constructor BaseTable(int, int, java.lang.Class<blah.blah.SomeTableEntry.class>)
    }
}

... where the genericised superclass is:

public class BaseTable<T extends BaseTableEntry>
{

    public BaseTable(int rows, int cols, Class<T> clasz)
    {
        ...
    }
...
}

I understand the compiler error, but cannot seem to find a workaround, other than to include an extra parameter in the SomeTable constructor.

Any suggestions?

Thanks!

Erubescence answered 25/1, 2010 at 3:53 Comment(2)
does the subclass also need to be generic? Or can it be SomeTable extends BaseTable<SomeTableEntry>?Camden
No, SomeTable does not need to be genericised, public class SomeTable extends BaseTable<SomeTableEntry> is fine.Erubescence
U
10

This compiles:

public class SomeTable extends BaseTable<SomeTableEntry> {
    public SomeTable(int rows, int cols)
    {
        super(rows, cols, SomeTableEntry.class);
    }
}

It works with a cast:

public class SomeTable<T extends BaseTableEntry> extends BaseTable<T> {
    public SomeTable(int rows, int cols)
    {
        super(rows, cols, (Class<T>)SomeTableEntry.class);
    }
}

but I'm looking forward to someone posting the explanation for why the compiler requires the cast for the class.

Upkeep answered 25/1, 2010 at 4:9 Comment(5)
+1. If he really needs SomeTable to be generic as well, I think he will have to duplicate the superclass constructor with the extra Class<T> parameter.Camden
@SteveB.: Just to make sure that the programmer knows what he/she is doing. super(rows, cols, (Class<T>)Object.class) will also work, but by doing a cast you are taking the responsibility`.Luhey
Okay, here is a stab at explanation. Since SomeTable is generic, he could declare another subclass SubTable extends SomeTable<SpecialEntry>. In this case, the hard-coded SomeTableEntry would not work anymore, because the expected erased type would now be SpecialEntry. So in conclusion, he must either make SubTable non-generic, or propagate the extra class parameter to the next generation.Camden
Actually, no need for another subclass at all. He can just write SomeTable<CompletelyUnrelatedBaseTableEntry> x = new SomeTable<CompletelyUnrelatedBaseTableEntry>(row,col) and be in trouble.Camden
I might throw something into the mix here: I found that if I changed my BaseTable constructor to public BaseTable(int rows, int cols, Class<? extends T> clasz) it works as well...Erubescence
A
4

It is possible to define the SomeTable constructor generically if you pass Class to it the same way as you do with the base class:

public class BaseTable<T extends BaseTableEntry>
{
    public BaseTable(int rows, int cols, Class<? extends T> clazz)
    {
        // ...
    }
}

class SomeTable<T extends BaseTableEntry>
extends BaseTable<T>
{
    public SomeTable(int rows, int cols, Class<? extends T> clazz)
    {
        super(rows, cols, clazz);
    }
}
Annuitant answered 25/1, 2010 at 5:23 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.