I'm generally very happy with the design of the Java language but today I came across a strange design choice. I don't know any platform where you can suggest changes to the Java language for version 9 or 10 and therefore I'm writing this issue here.
With Optionals you have a much better control over the behaviour if a value is null
.
Today I mapped a value in a stream to an Optional
.
I was really surprised, that findFirst
then gives me a Optional<Optional<T>>
instead of just Optional<T>
.
Where is the point in having Optional<Optional<T>>
?
For Java 9 and Streams this is solved with .flatMap(Optional::stream)
but this is only a work around and boilerplate code.
Wouldn't it be much better if Optionals generally wouldn't stack and always flats down?
Optional<Optional<T>> => Optional<T>
Optional.of(Optional.of("")) => Optional.of("")
Further explanation:
Working with Optionals you don't care if the sub optional was null or if the actual object was null
. It always would come down to Optional.empty()
.
String s = null;
Optional<String> opt = null;
Optional.of(s) => Optional.empty()
Optional.of(opt) => Optional.empty()
Also if you are working with optionals you are only interested in the object. So if you try to get the object you always have to double get and check the optional.
if(optionalOfOptional.isPresent()) {
optional = optionalOfOptional.get();
if(optional.isPresent()) {
objectOfInterest = optional.get();
}
}
I even see this being a source for errors since you would always need to check both optionals if you condition on your object being present.
if(optionalOfOptional.isPresent() && optionalOfOptional.get().isPresent()) {
...
}
Only checking the first optional could easily lead to mistakes.
In addition, why would it make sense to have a Optional that is null
in the first place? Isn't optional there to get rid of working with null
?
On the method level this could be solved for sure.
For example Optional.of
could look something like:
public static <T> Optional<T> of(T value) {
return value instanceof Optional ? value : new Optional<>(value);
}
On the type level this is probably not that easy.
I talk about this elsewhere that an
Optional<T>
should never be null. It violates whatOptional<T>
promises: getting rid ofNullPointerException
s.Therefore, I think it is a deficient design in the Optional API to allow
Optional<Optional<T>>
. The methods to get instances of Optional should behave better when given an Optional. But that being said, they can't. They could be accomplished if Optional.of was overloaded except for one slight issue.Since you can have a generic type T that is of type Optional, you can't have two Optional.of methods (one for Optional and one for T). Occasionally it would prevent some code from compiling if someone had the bright idea to have an
Optional<Optional<T>>
. Take the below code, it will fail to compile.Without forcing a language and syntax change on the entire language (forbidding a
Optional<Optional<T>>
) the automatic conversion is not an easy task.Java 8 in general I found took developers a while to grasp. When it first came out, I'd see
Map<String, Optional<T>>
whenMap<String, T>
would have sufficed or I did seeOptional<Optional<T>>
. I even sawMap<Optional<T>, Optional<Set<Optional<T>>>>
instead of simplyMap<T, Set<T>>
. As time progressed and people grappled with the language, I feel we learned to better manage these typing and get more sensible typing.I think it is unfortunate Java doesn't do the conversion automatically but for the poor soul who may need an
Optional<Optional<T>>
one day, I'm willing to use map, flatMap, orElse, and ofNullable every once in awhile to keep my typing sane. Using those methods may help with your particular quandaries.Edit:
I see, either missed it or an edit, that OP saw that
Stream<Optional<T>>.first
returns anOptional<Optional<T>>
. What OP intends to do affects a compromise. They could .map the stream to strip the interior Optional or they could .filter & .map the stream to strip out empty Optionals. One sad, extra line of code.