6

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??? #### ???

asked Aug 15, 2011 at 18:54
1
  • 2
    the answer depends almost entirely on what ThirdPartyClass and someMajicMethod are Commented Aug 15, 2011 at 18:57

3 Answers 3

6

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.

answered Aug 15, 2011 at 19:30

2 Comments

how about simply 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.
6

Perhaps Array.newInstance(..) is of interest for you

answered Aug 15, 2011 at 19:00

2 Comments

he's not creating an array anywhere in this method
@newacct no he is not, but at one point in his majic method he might want to. There are not that much possibilities if you take object serialization away. And this seems to me the point of interest to create this array by his generic type argument. Or do you think I am wrong with that assumption?
3
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?

answered Aug 15, 2011 at 19:01

5 Comments

It's a good thing that someMajicMethod returns an array, if it was a List, it would be more complicated. Square[] can be cast to Shape[], but not List<Square> to List<Shape>.
My reading of the question was that if you pass Triangle.class to someMajicMethod, then it is going to return a Triangle, not a Triangle[]
@Aaron Maybe Mohamed should change is method to someMethod(Class<? extends Shape[]> type,...).
Yes. I made it work by making someMethod take array type. So it went like this : someMethod(Class<? extends Shape[]> arrayType, File shapeFile) { Shape[] shapes = foo.someMajicMethod(arrayType,shapeFile) }
@Mohamed - Glad it worked for you! Don't forget to accept an answer if you solved your problem.

Your Answer

Draft saved
Draft discarded

Sign up or log in

Sign up using Google
Sign up using Email and Password

Post as a guest

Required, but never shown

Post as a guest

Required, but never shown

By clicking "Post Your Answer", you agree to our terms of service and acknowledge you have read our privacy policy.

Start asking to get answers

Find the answer to your question by asking.

Ask question

Explore related questions

See similar questions with these tags.