Wednesday, December 23, 2015

Cast Away

Java Generics are neat and all, but one situation I find them highly annoying goes like this:
  1. An interface IBar and an interface IFoo with a method:

    public Set<? extends IBar> getBars ();
     
  2. Bar is a concrete implementation of IBar, and Foo is a concrete implementation of IFoo that gives a covariant return for the same method:

    public Set<Bar> getBars ();
     
  3. Another method that declares it returns a Set<? extends IBar> where I know the Set actually holds instances of Bar.
     
  4. The implementation of Foo.getBars() calls this other method and intends to return the Set.
     
  5. The compiler balks because this is not acceptable:

    Set<? extends IBar>() bars = x();return (Set<Bar>) bars;
Indignantly I balk because this cast should be fine! But in reality, the nature of how Java Generics works does not allow downcasting of parameterized types. The bytecode has no knowledge of these types and no way to check them at runtime.

An easy hack would be to subvert the Java Generics compiler checks, such as:

Set<? extends IBar>() bars = x();
return (Set<Bar>)(Set<?>) bars;

For those who don't care about type-safety, I'm sure you will jump at that solution. For those who treasure type-safety, though, this option is highly unpalatable.

For a type-safe solution, a direct cast is not acceptable, so a copy with the correct type must be generated. To provide a single method that would satisfy this requirement for any Collection type took a bit of finagling of both Java Generics and reflection. This is what I finally came up with:

public static <T,
               U extends T,
               C extends Collection<? extends T>,
               D extends Collection<U>>
 D specializedCollection (C generalized, Class<? extends U> type) {
    try {
        D specialized = (D) generalized.getClass().newInstance();
        for (T item : generalized) {
            if (type.isInstance(item)) {
                specialized.add((U)item);
            }
            else {
                throw new ClassCastException("Element not castable to " + type.getClass() + ": " + item);
            }
        }
        return specialized;
    } catch (IllegalAccessException | InstantiationException e) {
        throw new IllegalArgumentException("Could not create a " + generalized.getClass().getName(), e);
    }

}
  • The U parameterized type had to be passed into the method because it cannot be discovered via the D type, or the C instance provided.
  • Collections are supposed to provide a public no-argument constructor, so the use of reflection to create the new instance is acceptable.
  • The cast of item from a type T to a type U is an acceptable cast because by the types declared, U is a subtype of T. However, this cast does not occur in the bytecode since parameterized types are deleted upon compilation.
  • In order to ensure type safety, Class.isInstance() is explicitly called using the U parameterized type in order to generate the expected ClassCastException.
Now I can maintain type safety with a single method call such as:

Set<? extends IBar>() bars = x();
return specializedCollection(bars, Bar.class);

Less efficient, sure, but now I am type safe!

This post is also available at Quantum Strides, and while you're there check out my Java Fundamentals course!

Monday, December 7, 2015

What I Learned From My Commodore 64

Today, retired to a box in my garage, my beloved Commodore 64 awaits my decision on its ultimate fate. That contraption, archaic by today's standards, changed my fate.

Engaging a child in any endeavor requires grabbing attention and stimulating the imagination. Whereas computers are ubiquitous today, 30 years ago they were novel. Remember how smart phones were rather uncommon a few years ago? And how everyone wanted one, first just to have one, then to use them for what they could do: games, videos, and everything ENTERTAINING! Now consider how most of those writing programs for smart phones are in general much younger than the average software engineer. It's not accident. These young whelps found their imaginations engaged by the new and exciting smart phones, so they dove in to learn more about them and to figure out how to make the phones do what they wanted them to do.

That was me 30 years ago. Does that make me an old fogey now? In some ways, I suppose, since my imagination did not get as an engaged with smart phones. That is partly due to my age, but also partly due to not having the time to indulge since I also have children and other activities that I actively pursue. But back then, I did.

I loved how I could play my favorite games on the Commodore 64, games like Jumpman, M.U.L.E., Olympic games, SportTime Soccer, and the amazing Alternate Reality series. Those certainly engaged my imagination. Then I discovered that BASIC was built right into the computer. I could write my own programs and make the computer do what I wanted it to do! Those small programs would mean nothing to anyone else, but they meant everything to me.

This is what I learned from my Commodore 64:
  • Save frequently.
  • Make backup copies.
  • A bug is not so bad, but if that bug ruins my work or fun then you and I are done.
  • Copy protection reduces customer value and satisfaction.
  • Plan for future change. Leave gaps between your line numbers.
  • Any program of significance cannot be kept entirely in your head.
  • REM only useful comments. Fix them if they are wrong.
  • A hole punch can do amazing things.
  • Know your variables.
  • Debug statements might be the only way to figure out what is going on.
  • Good graphics do not make a good game.
  • Good music can make a good game great!
  • Read other people's code.
  • Adjust to how other people write their code.
  • Adapt other people's techniques to your own style.
  • You can create any data structure in any programming language.
  • Memory is precious.
  • Be very careful where and what you poke.
  • If all else fails, power off and start all over.