Java switch statement: Constant expression required, but it IS constant

前端 未结 13 1439
予麋鹿
予麋鹿 2020-11-22 10:42

So, I am working on this class that has a few static constants:

public abstract class Foo {
    ...
    public static final int BAR;
    public static final          


        
相关标签:
13条回答
  • 2020-11-22 11:12

    I understand that the compiler needs the expression to be known at compile time to compile a switch, but why isn't Foo.BA_ constant?

    While they are constant from the perspective of any code that executes after the fields have been initialized, they are not a compile time constant in the sense required by the JLS; see §15.28 Constant Expressions for the specification of a constant expression1. This refers to §4.12.4 Final Variables which defines a "constant variable" as follows:

    We call a variable, of primitive type or type String, that is final and initialized with a compile-time constant expression (§15.28) a constant variable. Whether a variable is a constant variable or not may have implications with respect to class initialization (§12.4.1), binary compatibility (§13.1, §13.4.9) and definite assignment (§16).

    In your example, the Foo.BA* variables do not have initializers, and hence do not qualify as "constant variables". The fix is simple; change the Foo.BA* variable declarations to have initializers that are compile-time constant expressions.

    In other examples (where the initializers are already compile-time constant expressions), declaring the variable as final may be what is needed.

    You could change your code to use an enum rather than int constants, but that brings another couple of different restrictions:

    • You must include a default case, even if you have case for every known value of the enum; see Why is default required for a switch on an enum?
    • The case labels must all be explicit enum values, not expressions that evaluate to enum values.

    1 - The constant expression restrictions can be summarized as follows. Constant expressions a) can use primitive types and String only, b) allow primaries that are literals (apart from null) and constant variables only, c) allow constant expressions possibly parenthesised as subexpressions, d) allow operators except for assignment operators, ++, -- or instanceof, and e) allow type casts to primitive types or String only.

    Note that this doesn't include any form of method or lambda calls, new, .class. .length or array subscripting. Furthermore, any use of array values, enum values, values of primitive wrapper types, boxing and unboxing are all excluded because of a).

    0 讨论(0)
  • If you're using it in a switch case then you need to get the type of the enum even before you plug that value in the switch. For instance :

    SomeEnum someEnum = SomeEnum.values()[1];

    switch (someEnum) {
                case GRAPES:
                case BANANA: ...
    

    And the enum is like:

    public enum SomeEnum {
    
        GRAPES("Grapes", 0),
        BANANA("Banana", 1),
    
        private String typeName;
        private int typeId;
    
        SomeEnum(String typeName, int typeId){
            this.typeName = typeName;
            this.typeId = typeId;
        }
    }
    
    0 讨论(0)
  • 2020-11-22 11:19

    Got this error in Android while doing something like this:

     roleSpinner.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() {
            @Override
            public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {
    
                switch (parent.getItemAtPosition(position)) {
                    case ADMIN_CONSTANT: //Threw the error
    
                }
    

    despite declaring a constant:

    public static final String ADMIN_CONSTANT= "Admin";

    I resolved the issue by changing my code to this:

    roleSpinner.setOnItemSelectedListener(new AdapterView.OnItemSelectedListener() {
            @Override
            public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {
    
                String selectedItem = String.valueOf(parent.getItemAtPosition(position));
                switch (selectedItem) {
                    case ADMIN_CONSTANT:
    
                }
    
    0 讨论(0)
  • 2020-11-22 11:20

    I got this error on Android, and my solution was just to use:

    public static final int TAKE_PICTURE = 1;
    

    instead of

    public static int TAKE_PICTURE = 1;
    
    0 讨论(0)
  • 2020-11-22 11:20

    In my case, I was getting this exception because

    switch (tipoWebServ) {
                                case VariablesKmDialog.OBTENER_KM:
                                    resultObtenerKm(result);
                                    break;
                                case var.MODIFICAR_KM:
                                    resultModificarKm(result);
                                    break;
                            }
    

    in the second case I was calling the constant from the instance var.MODIFICAR_KM: but I should use VariablesKmDialog.OBTENER_KM directly from the class.

    0 讨论(0)
  • 2020-11-22 11:27

    You get Constant expression required because you left the values off your constants. Try:

    public abstract class Foo {
        ...
        public static final int BAR=0;
        public static final int BAZ=1;
        public static final int BAM=2;
        ...
    }
    
    0 讨论(0)
提交回复
热议问题