How to detect a new value was added to an enum and is not handled in a switch
Asked Answered
S

4

7

From time to time I have to add a new value to a enum type in my project.

public enum Day {
  SUNDAY, MONDAY, TUESDAY, WEDNESDAY, THURSDAY, FRIDAY, SATURDAY, 
   FILENOTFOUND //this one is new one
}

What I would like is to have a compile time error for every switch I have that is not treating the new value, like this one:

switch (color) {
        case MONDAY: 
        case TUESDAY: 
        case WEDNESDAY: 
        case THURSDAY: 
                    System.out.println("Mondays are bad.");
                     break;

        case FRIDAY: System.out.println("Fridays are better.");
                     break;

        case SATURDAY:
        case SUNDAY: System.out.println("Weekends are best.");
                     break;
    } 

Having a default: that throws some exception is not good enough, I would like it to be compile time.

I don't think this is possible but maybe someone has a neat trick...

I thought Findbugs would have a rule to find those but I only saw this: Eq: Covariant equals() method defined for enum (EQ_DONT_DEFINE_EQUALS_FOR_ENUM)

EDIT: I'm choosing Mark's reply, I do use Eclipse and that sounds just like what I needed! I am not an expert in findbugs at all so I might have missed such functionality, though I don't think so.

Sain answered 22/10, 2010 at 18:47 Comment(10)
add comment near enum declaration:-)Ezana
+1 for adding FILENOTFOUND to an unrelated set of enum constantsSuccess
+0 for adding FILENOTFOUND to an unrelated set of enum constants. Why do people still find this funny? (I don't mind the OP or anybody using it in an ad-hoc example, but honestly people still think this is funny and upvote accordingly, even after seeing this reference like 8000 times?)Felon
BTW, +1 for it being a good question.Felon
my + 1 was also for being a good question. Calm down and breathe.Success
I'm calm. I'm just losing faith in our collective sense of humour.Felon
This actually looks like using Polymorphism would be a better idea.Passionate
Potential duplicate: #16798029Offstage
the duplicate is probably the one asked 3 years later, don't you think?Sain
Possible duplicate of How to ensure completeness in an enum switch at compile time?Bessel
F
8

Eclipse has a compile-time warning/error you can enable: Enum constant not covered on "switch".

From your Project properties (or general preferences), go to Java Compiler->Errors/Warnings , check Enable project specific settings. You'll find the warning under Potential programming problems. It's set to Ignore by default but you can bump it up to Warning or Error.

Edit: I thought this goes without saying but I suppose I'll say it anyway: this is only applicable if you're developing in Eclipse or using it for your build management. Obviously a Findbugs or similar equivalent would be the "real" answer since it transcends the IDE and can be integrated into the build process.

Felon answered 22/10, 2010 at 18:52 Comment(8)
That's a nice feature but it won't help on existing switches on the enum you are adding a new value to.Ronnie
@rsp: Yes it does. If you go back and edit the enum, adding a new value, it will generate warnings/errors over all switch statements that use that enum.Felon
It does surprise me that Eclipse would have this while Findbugs doesn't.Felon
@Mark Peters, yes, for code in the same project, how about code in projects using your enum from a library.jar?Ronnie
@rsp: What about them? If they're compiled through eclipse, this will catch them. If they're not under your control, obviously you can't catch those errors but then neither will anything else. E.g. if it's a distributed library jar, do you expect to see error messages from all 800,000 clients that use your jar? Obviously you need control of your jar to find it. He asked for a compile error, which implies he's compiling it.Felon
@Mark Peters, my point is that an IDE feature does not make the cient code containing switches on enums robust against changes unless they are recompiled against the new version. It does help when the client code logs unhandled cases.Ronnie
@rsp: Sure, but that's not what the OP asked for. He asked for a compile-time error which obviously implies he's going to be recompiling the client code.Felon
I like to think we are also here to help people create better code, even if the advice is not what they asked for exactly.Ronnie
R
2

You could do that by adding a default clause and logging when it is visited:

switch (color) {
default:
    log.error("Unknown color in switch: " + color);
    break

case MONDAY: /*FALLTHROUGH*/
case TUESDAY: 

(adding fallthrough comments help later maintainers to decide whether you forgot code or not :-))

Edit Clarification copied from comments on Mark's answer:

An IDE feature signalling cases like this is fine for the developer at the moment of the change but it does not catch changes in other parts of code that your code depends on.

It does not make the cient code containing switches on enums robust against changes unless they are recompiled against the new version.

It does help when the client code logs unhandled cases; deployed code does not always have full control over its classpath or the versions of libraries on it.

Ronnie answered 22/10, 2010 at 18:53 Comment(5)
this is a compile-time check as the OP requested?Iene
Not compile-time so limited in use and unapplicable to the question. Edit: removed d/v, this is still useful so I can't call it "wrong".Felon
@Kirk Woll, no it isn't. Using this pattern makes your code more robust against changes to used classes/enums even if you don't recompile the code containing the switch. As I see it is is a better alternative to make sure you don't miss this kind of situation even if your IDE does support this type of check.Ronnie
If runtime is the route this might be a good candidate for an assert or similar; something that can be removed when building production code.Felon
@Mark, part of my point is that checks like these should be present in production code, not just during the development / test / acceptance phases.Ronnie
M
0

IntelliJ was a check where not all case have been set. It has an auto-fix to fill in the missing cases.

Mccaslin answered 22/10, 2010 at 20:39 Comment(0)
B
0

Since Java 14 (with JEP 361) there are now Switch Expressions which lets you do that. For your example this would then look like that:

String feeling = switch (day) {
    case MONDAY => "Mondays are bad.";
    case TUESDAY => "Tuesdays are bad.";
    case WEDNESDAY => "Wednesdays are bad.";
    case THURSDAY => "Thursdays are bad.";
    case FRIDAY => "Fridays are better.";
    case SATURDAY => "Weekends are best.";
    case SUNDAY => "Weekends are best."
}
System.out.println(feeling);

If you add a new enum value FILENOTFOUND then you get a compile error: java: the switch expression does not cover all possible input values.

Beneficial answered 13/9, 2023 at 15:32 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.