在Java中,常量是可以在final class里的,并不仅仅只是存在接口中。事实上,Java常量的使用一般有三种方式:

1.接口中定义常量

public interface Constants {
    int SEX_MALE = 1;
    int SEX_FEMAL = 2;
}

2.类中定义常量

public class Constants {
    public static final int SEX_MALE = 1;
    public static final int SEX_FEMAL = 2;
}

3.枚举

public enum Constants {
    SEX_MALE(1, "男"),SEX_FEMALE(2, "女");
    
    private int sex;
    private String field;
    
    Constants(int sex, String field){
        this.sex = sex;
        this.field = field;
    };

}

这三种方式其实都可以,也能在各个项目中看到这三种方式的身影,不存在什么一定要怎样怎样的说法。不过估计这回答并不是你想要的。如果要深究常量到底存放在哪好的话,不管什么类型,其创建的目的只是为了维护常量的话,其实是不推荐使用接口的。尽管在接口中定义常量显得更简洁,因为不用写public static final了嘛,但在《Effective Java》这本书中提到过:

如果某个实现了常量接口的类被修改不再需要常量了,也会因为序列化兼容原因不得不保持该实现,而且非final类实现常量接口会导致所有子类被污染

同时书中也说到,接口应当只用于定义类型,不应该只用于导出常量。

那么另外两种方式哪个更“好”呢?这就要看使用场景了。

如果是简单的共有常量定义,则强烈推荐枚举,比如上面代码演示的,只是简单的常量标识。

如果是复杂的常量,则可以放在常量类里管理:

// class的类型为final,防止该类被继承
public final class Constants {
    // 私有构造函数,避免该类被实例化
    private Constants() {}
    
    public static final String URL = "https://segmentfault.com/";
}

RudeCrab
22 声望2 粉丝