I have an ArrayList with a certain class (Piece) from which I want to pick items with a subclass (Miner and Bomb). However, when I pick some items it will use the methods of the super class (Piece), and not it's subclasses.
Since ArrayList<Pieces>
will contain several subclass objects of pieces, it is not possible to define a more specific class for ArrayList. Is it possible to pick some elements from ArrayList and still use the methods from the subclasses?
(I found it a little hard to find the solution for this simple problem. I searched for overriding, arrayList, methods along some other things, but couldn't find it. If it was too obvious to found, I'd like to know where you found it.)
Class Main:
import java.util.ArrayList;
public class Main {
public static void main(String[] args) {
ArrayList<Piece> pieceList = new ArrayList<>();
pieceList.add(new Miner());
pieceList.add(new Bomb());
// create new classes to test canDefeat
if (new Miner().canDefeat(new Bomb())) {
System.out.println("Yes! This line will be printed.");
}
System.out.println();
// test canDefeat with pieces from the pieceList
if (pieceList.get(0).canDefeat(pieceList.get(1))) {
System.out.println("No, this line will not be printed");
}
}
}
Class Piece:
public abstract class Piece {
public boolean canDefeat(Piece opponent) {
return false;
}
}
Class Miner:
public class Miner extends Piece {
public boolean canDefeat(Bomb opponent) {
return true;
}
}
Class Bomb:
public class Bomb extends Piece {
}
The why of this is easy:
Your
Miner
doesn't overridecanDefeat(Piece)
, it adds a new method with the signaturecanDefeat(Bomb)
.In the code
pieceList.get(0).canDefeat(pieceList.get(1))
, all the compiler has to work with is the typePiece
, and so it resolves that toPiece#canDefeat(Piece)
.The what to do instead is more tricky.
I would suggest that what kinds of pieces can defeat what other kinds of pieces isn't an attribute of each piece; for one thing, that will get brittle fairly quickly if you change the rules.
Instead, I would look at alternatives like:
Perhaps each piece can have a strength level, where higher strength means the piece can defeat a piece with lower strength. (Example below.)
You could have a
BattleLogic
class that can compare types of pieces to determine the outcome, which would either work with strength per the above, or by type usinginstanceof
, etc. Normally, when you reach forinstanceof
you need to stop and think about what you're doing and whether it's really the right tool for the job (it usually isn't), but it may well be the right tool for this job. The advantage to aBattleLogic
class is that none of the pieces knows anything about the other pieces; instead, you have one thing that understands the game as a whole and how the pieces rank when they fight. (Example below.)If you want each
Piece
to know what other types of pieces it can defeat, you can do that, but I think it will tend to lead to an unmaintainable mess. Two ways come to mind:Having a list of the types it can defeat. (Example below.)
Using reflection to achieve what you were trying to do originally. (Example below.)
Here's the strength suggestion in code:
And the
BattleLogic
suggestion in code:Here's that third suggestion, having each type of piece know what pieces it can defeat, in case. Again, though, I don't suggest this; I think having the pieces know this much about each other is going to be a maintenance problem:
Or you could make the list static if you're going to have millions of pieces and memory is an issue; this is just a sketch.
And finally, the way using reflection: