But a few things make me gnash my teeth on a daily basis, and it's funny, but none of them are issues in another JVM language in which I have been dabbling, Scala. It seems the developers of that language felt my pain as well.
- Miserable type inference. Apparently some of the problems with it are being addressed in project coin for Java 7. The blue portions of the following code are, to any sane programmer, maddeningly superfluous, but nevertheless strictly required in Java until at least mid-2011:
List< Integer > li1 = new ArrayList< Integer >();
List< Integer > li2 = Arrays.asList( 1, 2, 3 );
o.processListOfInteger( Arrays.< Integer >asList() );
Needless to say, equivalent initializations in Scala require no such redundant information. - Generic invariance. An example from last week: I'm working on an implementation
FrazzleExecutorService
ofjava.util.concurrent.ScheduledExecutorService
and a refinementFrazzleFuture< T >
ofjava.util.ScheduledFuture< T >
. Covariant subtyping lets me get away with returning aFrazzleFuture< T >
from a method likeFrazzleExecutorService.submit()
without violating the contract. But I can't returnList< FrazzleFuture< T > >
fromFrazzleExecutorService.invokeAll()
because (a)ScheduledExecutorService
would have had to declare the return type to beList< ? extends ScheduledFuture< T > >
; (b) the returned list should have been immutable anyway; (c) generic types likeList< T >
are invariant in their parameters. In Scala, there is a separate mutable and immutable collections hierarchy, and at least in the immutable one,S
<:T
impliesList[ S ] <: List[ T ]
, becauseList
is declared covariant in its parameter. - Collections can't be tersely initialized. Part of the blame is the Collections framework; part of it is the goddamn language. The following code illustrates, with green text indicating typical verbosity:
List< Integer > li1 = new ArrayList< Integer >(
Arrays.asList( 1, 2, 3 )
);
@SuppressWarnings( "serial" )
Map< String, String > mss1 = new HashMap< String, String >() { {
put( "foo", "bar" );
put( "this", "sucks" );
} };
Turns out the collections literals are also not supported in Scala; you have to typeList( 1, 2, 3 )
orMap( "foo" -> "bar", "this" -> "rocks" )
. Excuse me if I mock Java incessantly at this point. Collection improvements have been postponed until Java 8, scheduled for mid-2012.
- Higher-order programming is absurdly verbose. Rather than give code samples, I'll just refer you to these guys and let you see for yourself how even a library can't save you from massive boilerplate for the simplest things. And Scala? Lambda expressions are built-in as syntactic sugar for functional objects, making higher-order code simple, readable, and terse.
- Modeling variant types is awkward. You have to choose from among many bad options:
Scala has case classes and pattern matching built in.Representation Interrogation one sparsely-populated class (S + T modeled as S × T) if
-ladders based on comparing tonull
(see 8)S × T and an enum of type labels switch + casting a hierarchy a bunch of isS()
andisT()
methods and castinga hierarchy various casting attempts wrapped with ClassCastException
catch blocks (ok, that's not really an option, but I get that as an answer in interviews sometimes)a hierarchy if
-ladders based oninstanceof
and castinga hierarchy polymorphic decomposition and the inevitable bloated APIs at the base class that result a hierarchy that includes Visitor painfully verbose visitors (see 4 and 10) a hierarchy of Throwable
sthrow and various catch
blocks, which I suspect compiles to the same thing as the instanceof approach, only more expensive (but actually requires the least code!) - No tuples. One ends up either creating
Pair< S, T >
or dozens of throwaway classes with "And" in the name, likeCountAndElapsed
. Scala has tuples, although I feel like they kind of screwed up by not going the full ML and making multi-argument functions/methods be the same as single-argument functions/methods over tuples. So to call a 2-arg functionf
with a pairp = ( p1, p2 )
, you can either callf( p1, p2 )
orf.tupled( p )
. There must be some deep reason for making the distinction. - No mixins. If you need stuff from 2 abstract classes, you will be copying, or aggregating (with loads of monkey delegation boilerplate) at least one of the two.
- Null. The following code should illustrate:
private static Doohickey getDoohickey( Thingamajigger t ) {
Whatsit w;
Foobar f;
if ( null == t )
return null;
else if ( null == ( w = t.getWhatsit() ) )
return null;
else if ( null == ( f = w.getFoobar() ) )
return null;
else
return f.getDoohickey();
}
I believe the "Elvis" operator was developed to solve this annoyance (return t.?getWhatsit().?getFoobar().?getDoohickey();
) but it did not make the cut for Java 7 or even Java 8, from what I understand. Scala's solution to this issue is to recommend that operations which might not have a value for you returnOption[ T ]
instead ofT
. You can then map your method call to the Option and get back another Option without ever seeing a null pointer exception. Option is a variant type, easily modeled in Scala but not in Java (see 5).
- Iterators. They can't throw checked exceptions. They have to implement
remove()
, often by throwing (unchecked)UnsupportedOperationException
s. For-each syntax can't work with iterators directly. None of these problems arise with the superb collections framework in Scala which is designed hand-in-hand with its clean higher-order programming (see 4). - Void. This is a holdover from C, and is obviously not anything Java can get rid of, but it's stupid. Because of
void
, e.g., I can never do a visitor pattern with just one kind of visitor; there has to be one whose methods return a generic type T, and another whose methods returnvoid
. And don't try to sell me on the psuedo-typeVoid
, because you still have to accept or returnnull
somewhere. Scala has a typeUnit
with a single trivial value()
, and unitary methods/functions can explicitly return that value or not return anything; the semantics are the same. Thus all expressions have some meaningful type, and classes with generic types can be fully general.