所有分类
  • 所有分类
  • 未分类

Java之枚举(enum)-使用/教程/实例

简介

说明

本文用示例介绍Java中的枚举(enum)的用法。

使用场景

定义常量、状态机等。

枚举与常量类

常量类

以往设置常量,通常将常量放置在接口中,这样在程序中就可以直接使用了,并且该常量不能被修改,因为在接口中定义的常量时,该常量的修饰符为final与static。如:

public interface IConstants{
	public static final int RED = 1;
	public static final int BLUE = 2;
	public static final int GREEN = 3;
}

枚举与常量类的对比

枚举类常量类
单例完全单例、线程安全。 枚举类编译后类为:public final class T extends Enum,不允许继承可防止被子类修改。不单例 常量类可被继承修改、增加字段等,容易导致父类的不兼容。
性能性能高。 常量值地址唯一,可以用==直接对比性能低 使用常量类时,往往得通过equals去判断两者是否相等。
引用类不需重新编译引用类。 枚举类编译时,没有把常量值编译到代码里,即使常量的值发生变化,也不会影响引用常量的类。需要重新编译引用类。 常量类编译时,是直接把常量的值编译到类的二进制代码里,常量的值在升级中变化后,需要重新编译引用常量的类,因为里面存的是旧值。
越界不会越界。 编译期间限定类型,不允许发生越界的情况。可能越界。 switch语句支持枚举型,当switch使用int、String类型时,由于值的不稳定性往往会有越界的现象,对于这个的处理往往只能通过if条件筛选以及default模块来处理。

枚举与普通类

差别不大。枚举也可以定义变量与方法:

package com.example.a;

enum MyEnum{
    FIRST("第一个"),
    SECOND("第二个"),
    ;

    private final String description;
    
    MyEnum(String description) {
        this.description = description;
    }

    public String getDescription() {
        return description;
    }

    private String lastName;

    public String getLastName() {
        return lastName;
    }

    public void setLastName(String lastName) {
        this.lastName = lastName;
    }
}

public class Demo {
    public static void main(String[] args) {
        MyEnum.FIRST.setLastName("Tony");
        System.out.println(MyEnum.FIRST.getLastName());
        System.out.println(MyEnum.FIRST.getDescription());
    }
}

执行结果

Tony
第一个

枚举与数据库

枚举型可以直接与数据库打交道,我通常使用varchar类型存储,对应的是枚举项的名字。

在下边的例子中,如果某个类中的字段为:private PersonType type; 则其对应的数据库中的值为:”STUDENT”、”TEACHER”。

package com.example.a;

public enum PersonType{
    STUDENT("学生"),
    TEACHER("教师"),
    ;

    private final String description;

    PersonType(String description) {
        this.description = description;
    }

    public String getDesc() {
        return description;
    }
}

基础示例

一个参数

实例 

package com.example.a;

enum PayTypeEnum{
    ALIPAY("支付宝"),
    WECHAT("微信"),
    UNION_PAY("银联"),
    ;

    //这个必须定义,且成员变量的类型及个数必须对应于上边枚举的定义
    //枚举标识码(英文描述)
    private String description;

    //必须提供为私有的,防止外部new对象
    PayTypeEnum(String description) {
        this.description = description;
    }

    public String getDescription() {
        return description;
    }
}

public class Demo {
    public static void main(String[] args){
        for (PayTypeEnum value : PayTypeEnum.values()) {
            System.out.println("name:" + value.name() + "," + "description:" + value.getDescription());
//            下边这样写结果是一样的
//            System.out.println("name:" + value.toString() + "," + "description:" + value.getDescription());
        }
    }
}

运行结果

name:ALIPAY,desc:支付宝
name:WECHAT,desc:微信
name:UNIONPAY,desc:银联

多个参数

实例 

package com.example.a;

interface IEnum{
    String getCode();
    String getDescription();
}

enum PayTypeEnum implements IEnum{
    ALIPAY("ALIPAY", "支付宝"),
    WECHAT("WECHAT", "微信"),
    UNION_PAY("UNION_PAY", "银联"),
    ;

    //这个必须定义,且成员变量的类型及个数必须对应于上边枚举的定义
    //枚举标识码(英文描述)
    private final String code;

    //枚举标识码(中文描述)
    private final String description;

    //必须提供,而且是私有的,防止外部new对象
    PayTypeEnum(String code, String description) {
        this.code = code;
        this.description = description;
    }

    public static String getDescriptionByCode(String code){
        for(PayTypeEnum value : PayTypeEnum.values()){
            if(value.getCode().equals(code)){
                return value.getDescription();
            }
        }
        return null;
    }

    @Override
    public String getCode() {
        return this.code;
    }

    @Override
    public String getDescription() {
        return this.description;
    }
}

public class Demo {
    public static void main(String[] args){
        System.out.println(PayTypeEnum.getDescriptionByCode(PayTypeEnum.ALIPAY.getCode()));
        System.out.println(PayTypeEnum.getDescriptionByCode(PayTypeEnum.WECHAT.getCode()));
    }
}

运行结果

支付宝
微信

异常处理应用

异常码枚举

package enums.expection;

import org.apache.commons.lang.StringUtils;

public enum ErrorCodeEnum {
    SYS_ERROR(1001, "系统错误,请重试"),
    UNKNOWN_ERROR(1002, "未知的系统异常"),
    SERVICE_INVOKE_FAIL(1003, "服务调用失败"),
    ILLEGAL_ARGS(1004, "参数校验错误"),
    ;
    

    /**
     * 结果码值
     */
    private final Integer code;
    
    /**
     * 描述
     */
    private final String description;

    ErrorCodeEnum(Integer code, String description) {
        this.code = code;
        this.description = description;
    }

    public static ErrorCodeEnum getByValue(String code) {
        for (ErrorCodeEnum result : values()) {
            System.out.println(result.ordinal());
            if (StringUtils.equals(result.getCode(), code)) {
                return result;
            }
        }
        return null;
    }

    public Integer getCode() {
        return code;
    }

    public String getDescription() {
        return description;
    }

}

自定义异常

package enums.expection;

import org.apache.commons.lang.StringUtils;

import java.util.HashMap;
import java.util.Map;

public  class WangException extends RuntimeException {

    /** 错误码枚举*/
    private ErrorCodeEnum errorCode;

    /**
     * 详细错误信息
     */
    private Map<String, String> errorMap = new HashMap<String, String>();

    /**
     * 带参构造器
     */
    public WangException(ErrorCodeEnum errorCode) {
        super(errorCode.getDesc());
        this.setErrorCode(errorCode);
    }

    /**
     * 带参构造器.
     */
    public WangException(ErrorCodeEnum errorCode, String message) {
        super(StringUtils.isNotBlank(message) ? message : errorCode.getDesc());
        this.setErrorCode(errorCode);
    }

    /**
     * 带参构造器
     */
    public WangException(ErrorCodeEnum errorCode, Map<String, String> errorMap) {
        this(errorCode);
        this.errorMap = errorMap;
    }
    /**
     * 带参构造器
     */
    public WangException(String message) {
        super(message);
        this.setErrorCode(ErrorCodeEnum.UNKNOWN_ERROR);
    }

    /**
     * Gets error code.
     */
    public ErrorCodeEnum getErrorCode() {
        return errorCode;
    }
    
    /**
     * Sets error code.
     */
    public void setErrorCode(ErrorCodeEnum errorCode) {
        this.errorCode = errorCode;
    }
    /**
     * Gets error map.
     */
    public Map<String, String> getErrorMap() {
        return errorMap;
    }
    
    /**
     * Sets error map.
     */
    public void setErrorMap(Map<String, String> errorMap) {
        this.errorMap = errorMap;
    }

    private static String findMessage(Map<String, String> errorMap) {
        if (errorMap.isEmpty()) {
            return null;
        }
        return errorMap.values().iterator().next();
    }

}

测试类

package enums.expection;

public class Test {

    public static void main(String[] args) {
        String name="";
        int i=0;
        try {
            if (name == null)
                throw new WangException(ErrorCodeEnum.ILLEGAL_ARGS);
            if(i==0)
                throw new WangException(ErrorCodeEnum.ILLEGAL_ARGS, "参数不能为0");
        }catch (WangException e){
            e.printStackTrace();
            System.out.println("异常码:"+e.getErrorCode().getCode());
            System.out.println("异常描述:"+e.getMessage());
        }
    }
}

高级操作

接口中定义注解

package org.example.a;

interface Color {
    enum Green {LIGHT_GREEN, DARK_GREEN}
    enum Red {LIGHT_RED, DARK_RED}
}

public class Demo {
    public static void main(String[] args) {
        System.out.println(Color.Green.DARK_GREEN.ordinal());
    }
}

执行结果

1

枚举上使用注解

见:Java之注解-使用/实例 – 自学精灵

实现原理

枚举类

package org.example.a;

public enum Season {
    SPRING, SUMMER, AUTUMN, WINTER;
}

反编译代码

public final class Season extends Enum
{
    private Season(String s, int i)
    {
        super(s, i);
    }
    public static Season[] values()
    {
        Season at[];
        int i;
        Season at1[];
        System.arraycopy(at = ENUM$VALUES, 0, at1 = new Season[i = at.length], 0, i);
        return at1;
    }

    public static Season valueOf(String s)
    {
        return (Season)Enum.valueOf(demo/Season, s);
    }

    public static final Season SPRING;
    public static final Season SUMMER;
    public static final Season AUTUMN;
    public static final Season WINTER;
    private static final Season ENUM$VALUES[];
    static
    {
        SPRING = new Season("SPRING", 0);
        SUMMER = new Season("SUMMER", 1);
        AUTUMN = new Season("AUTUMN", 2);
        WINTER = new Season("WINTER", 3);
        ENUM$VALUES = (new Season[] {
            SPRING, SUMMER, AUTUMN, WINTER
        });
    }
}

分析

通过反编译后代码可见,该类继承了Enum类,同时final关键字表明这个类不能被继承。

字段都是static类型的,static类型的属性会在类被加载之后被初始化。当一个Java类第一次被真正使用到的时候静态资源被初始化、Java类的加载和初始化过程都是线程安全的。所以,创建一个enum类型是线程安全的。 ​

0

评论0

请先

显示验证码
没有账号?注册  忘记密码?

社交账号快速登录