I have a hierarchy where Square, Triangle and Circle all extend from Shape. I have a working method:
public void someMethod() {
File file = new File("File_with_squares");
ThirdPartyClass foo = new ThirdPartyClass();
Square[] squares = foo.someMajicMethod(Square[].class,file);
for (Square square: squares)
square.draw();
}
Now I want to make this method generic so that it can accept any shape. I want to be able to call it someMethod(Triangle.class,new File("File_with_triangles")
or someMethod(Circle.class, new File("File_with_circles")
. I am trying like this:
public void someMethod(Class<? extends Shape> type, File shapeFile) {
ThirdPartyClass foo = new ThirdPartyClass();
#### What goes here??? ####
for (Shape shape: shapes)
shape.draw();
}
What should be there at #### What goes here??? #### ???
3 Answers 3
Assuming ThirdPartClass.someMajicMethod has a signature something like this:
public <T> T someMajicMethod(Class<T> class1, File file);
Then you should be able to do something like this:
public void someMethod(Class<? extends Shape> type, File shapeFile) {
ThirdPartyClass foo = new ThirdPartyClass();
@SuppressWarnings("unchecked")
Class<? extends Shape[]> arrayType =
(Class<? extends Shape[]>) Array.newInstance(type, 0).getClass();
assert Shape[].class.isAssignableFrom(arrayType);
Shape[] shapes = foo.someMajicMethod(arrayType, shapeFile);
for (Shape shape: shapes)
shape.draw();
}
So if you call someMethod(Triangle.class, file)
, then arrayType
will be Triangle[].class
in the call to someMajicMethod
.
Though you may find it simpler to have someMethod take the array type as a parameter instead of the element type so you can avoid that step.
2 Comments
assert Shape.class.isAssignableFrom(type)
?type
's type is already enforced by the compiler. I'm asserting arrayType
's type because we've just done an unchecked cast (the generic type info is going to be erased at compile time). And also, asserts in Java won't execute unless you enable them explicitly in the VM. Asserts are better used as documentation than as enforcement. In this case, I can guarantee that that assert will alway pass unless Array.newInstance
is broken.Perhaps Array.newInstance(..) is of interest for you
2 Comments
Shape[] shapes = foo.someMajicMethod(type, file);
If foo
is a third-party class, I assume you don't control the API of it. I assumed it has the appropriate method signature to handle the line I've written but there is no way for me to be certain without more information about that class.
If this doesn't work, what is the problem?
5 Comments
Triangle.class
to someMajicMethod, then it is going to return a Triangle, not a Triangle[]someMethod(Class<? extends Shape[]> type,...)
.
ThirdPartyClass
andsomeMajicMethod
are