Steven Roose
Steven Roose

Reputation: 2779

Why can't I create an enum in an inner class in Java?

What I try to do is this:

public class History {
    public class State {
        public enum StateType {

Eclipse gives me this compile error on StateType: The member enum StateType must be defined inside a static member type.

The error disappears when I make the State class static. I could make State static, but I don't understand why I cannot declare an enum in an inner class.

Upvotes: 76

Views: 54536

Answers (4)

Joachim Sauer
Joachim Sauer

Reputation: 308259

This is allowed in Java 16+

This is no longer forbidden starting with Java 16 (and the later LTS release Java 17) thanks to JEP 395 relaxing the rules on some static types inside inner classes.

The new phrasing in §8.1.3. Inner Classes and Enclosing Instances is (emphsis mine):

All of the rules that apply to nested classes apply to inner classes. In particular, an inner class may declare and inherit static members (§8.2), and declare static initializers (§8.7), even though the inner class itself is not static.

This answer explains the change in more detail.

Before Java 16 it was forbidden:

enum types that are defined as nested types are always implicitly static (see JLS §8.9. Enums):

A nested enum type is implicitly static.

You can't have a static nested type inside a non-static one (a.k.a an "inner class", see JLS §8.1.3. Inner Classes and Enclosing Instances):

It is a compile-time error if an inner class declares a member that is explicitly or implicitly static, unless the member is a constant variable

Therefore you can't have an enum inner type inside a non-static nested type.

Upvotes: 102

K F
K F

Reputation: 1546

This worked for my use-case:

public class History {

    public interface HConstants{
         public enum StateType { PAST,CURRENT,FUTURE}
    }

    //Inner class
    public class State implements HConstants{
        public StateType stateField = StateType.PAST;

Upvotes: 3

Edwin Dalorzo
Edwin Dalorzo

Reputation: 78649

If you declared an enum like this:

enum Suit {SPADES, HEARTS, CLUBS, DIAMONDS}

The Java compiler would synthetically generate the following class for you:

final class Suit extends java.lang.Enum<Suit> {
  public static final Suit SPADES;
  public static final Suit HEARTS;
  public static final Suit CLUBS;
  public static final Suit DIAMONDS;
  private static final Suit[] $VALUES;
  public static Suit[] values();
  public static Suit valueOf(java.lang.String);
  private Suit();
}

There is no intention to create other instances of this class other than those static fields already defined in it (as you could infer from its private constructor), but most importantly, and as mentioned in the accepted answer, a inner class cannot have static members (JLS §8.1.3. Inner Classes and Enclosing Instances), and since the enum synthetic class does, it makes it unacceptable as inner class.

Upvotes: 20

Dhiral Pandya
Dhiral Pandya

Reputation: 10619

Already enough information from +Joachim Sauer, I am just adding some extra details.

You can define inner enum only if your inner class is static nested inner class. See below

private static class DbResource {

    public enum DB {
        MERGE_FROM, MERGE_TO, MAIN;
    }
}

Upvotes: 6

Related Questions