Skip to main content

Java Enum definition


I thought I understood Java generics pretty well, but then I came across the following in java.lang




Class Enum<E extends Enum<E>>



Could someone explain how to interpret this type parameter? Bonus points for providing other examples of where a similar type parameter could be used.



Cheers, Don.


Source: Tips4allCCNA FINAL EXAM

Comments

  1. It means that the type argument for enum has to derive from an enum which itself has the same type argument. How can this happen? By making the type argument the new type itself. So if I've got an enum called StatusCode, it would be equivalent to:

    public class StatusCode extends Enum<StatusCode>


    Now if you check the constraints, we've got Enum<StatusCode> - so E=StatusCode. Let's check: does E extend Enum<StatusCode>? Yes! We're okay.

    You may well be asking yourself what the point of this is :) Well, it means that the API for Enum can refer to itself - for instance, being able to say that Enum<E> implements Comparable<E>. The base class is able to do the comparisons (in the case of enums) but it can make sure that it only compares the right kind of enums with each other.

    I've used something similar in my C# port of ProtocolBuffers. There are "messages" (immutable) and "builders" (mutable, used to build a message) - and they come as pairs of types. The interfaces involved are:

    public interface IBuilder<TMessage, TBuilder>
    where TMessage : IMessage<TMessage, TBuilder>
    where TBuilder : IBuilder<TMessage, TBuilder>

    public interface IMessage<TMessage, TBuilder>
    where TMessage : IMessage<TMessage, TBuilder>
    where TBuilder : IBuilder<TMessage, TBuilder>


    This means that from a message you can get an appropriate builder (e.g. to take a copy of a message and change some bits) and from a builder you can get an appropriate message when you've finished building it. It's a good job users of the API don't need to actually care about this though - it's horrendously complicated, and took several iterations to get to where it is.

    ReplyDelete
  2. Here's the explanation I like best: Groking Enum (aka Enum<E extends Enum<E>>)

    ReplyDelete
  3. The following is a modified version of the explanation from the book Java Generics and Collections:
    We have an Enum declared

    enum Season { WINTER, SPRING, SUMMER, FALL }


    which will be expanded to a class

    final class Season extends ...


    where ... is to be the somehow-parameterised base class for Enums. Let's work
    out what that has to be. Well, one of the requirements for Season is that it should implement Comparable<Season>. So we're going to need

    Season extends ... implements Comparable<Season>


    What could you use for ... that would allow this to work? Given that it has to be a parameterisation of Enum, the only choice is Enum<Season>, so that you can have:

    Season extends Enum<Season>
    Enum<Season> implements Comparable<Season>


    So Enum is parameterised on types like Season. Abstract from Season and
    you get that the parameter of Enum is types that satisfy

    E extends Enum<E>




    Maurice Naftalin (co-author, Java Generics and Collections)

    ReplyDelete
  4. You are not the only one wondering what that means; see Chaotic Java blog.

    “If a class extends this class, it should pass a parameter E. The parameter E’s bounds are for a class which extends this class with the same parameter E”.

    ReplyDelete
  5. This post has totally clarified to me these problem of 'recursive generic types'.
    I just wanted to add another case where this particular structure is necessary.

    Suppose you have generic nodes in a generic graph:

    public abstract class Node<T extends Node<T>>
    {
    public void addNeighbor(T);

    public void addNeighbors(Collection<? extends T> nodes);

    public Collection<T> getNeighbor();
    }


    Then you can have graphs of specialized types:

    public class City extends Node<City>
    {
    public void addNeighbor(City){...}

    public void addNeighbors(Collection<? extends City> nodes){...}

    public Collection<City> getNeighbor(){...}
    }

    ReplyDelete

Post a Comment

Popular posts from this blog

[韓日関係] 首相含む大幅な内閣改造の可能性…早ければ来月10日ごろ=韓国

div not scrolling properly with slimScroll plugin

I am using the slimScroll plugin for jQuery by Piotr Rochala Which is a great plugin for nice scrollbars on most browsers but I am stuck because I am using it for a chat box and whenever the user appends new text to the boxit does scroll using the .scrollTop() method however the plugin's scrollbar doesnt scroll with it and when the user wants to look though the chat history it will start scrolling from near the top. I have made a quick demo of my situation http://jsfiddle.net/DY9CT/2/ Does anyone know how to solve this problem?

Why does this javascript based printing cause Safari to refresh the page?

The page I am working on has a javascript function executed to print parts of the page. For some reason, printing in Safari, causes the window to somehow update. I say somehow, because it does not really refresh as in reload the page, but rather it starts the "rendering" of the page from start, i.e. scroll to top, flash animations start from 0, and so forth. The effect is reproduced by this fiddle: http://jsfiddle.net/fYmnB/ Clicking the print button and finishing or cancelling a print in Safari causes the screen to "go white" for a sec, which in my real website manifests itself as something "like" a reload. While running print button with, let's say, Firefox, just opens and closes the print dialogue without affecting the fiddle page in any way. Is there something with my way of calling the browsers print method that causes this, or how can it be explained - and preferably, avoided? P.S.: On my real site the same occurs with Chrome. In the ex