• 枚举&包装类


    枚举

    某些类型的对象是有限的几个,这样的例子举不胜举:

    • 星期:Monday(星期一)…Sunday(星期天)
    • 性别:Man(男)、Woman(女)
    • 月份:January(1月)…December(12月)
    • 季节:Spring(春天)…Winter(冬天)
    • 支付方式:Cash(现金)、WeChatPay(微信)、Alipay(支付宝)、BankCard(银行卡)、CreditCard(信用卡)
    • 员工工作状态:Busy(忙)、Free(闲)、Vocation(休假)
    • 订单状态:Nonpayment(未付款)、Paid(已付款)、Fulfilled(已配货)、Delivered(已发货)、Checked(已确认收货)、Return(退货)、Exchange(换货)、Cancel(取消)

    枚举类型本质上也是一种类,只不过是这个类的对象是固定的几个,而不能随意让用户创建。

    在JDK1.5之前,需要程序员自己通过特殊的方式来定义枚举类型。

    在JDK1.5之后,Java支持enum关键字来快速的定义枚举类型。

    JDK1.5之前

    在JDK1.5之前声明枚举类

    • 构造器加private私有化
    • 本类内部创建一组常量对象,并添加public static修饰符,对外暴露这些常量对象

    示例代码:

    public class TestEnum {
    	public static void main(String[] args) {
    		Season spring = Season.SPRING;
    		System.out.println(spring);
    	}
    }
    class Season{
    	public static final Season SPRING = new Season();
    	public static final Season SUMMER = new Season();
    	public static final Season AUTUMN = new Season();
    	public static final Season WINTER = new Season();
    	
    	private Season(){
    		
    	}
    	
    	public String toString(){
    		if(this == SPRING){
    			return "春";
    		}else if(this == SUMMER){
    			return "夏";
    		}else if(this == AUTUMN){
    			return "秋";
    		}else{
    			return "冬";
    		}
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28

    JDK1.5之后

    语法格式:

    【修饰符】 enum 枚举类名{
        常量对象列表
    }
    
    【修饰符】 enum 枚举类名{
        常量对象列表;
        
        其他成员列表;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    示例代码:

    public class TestEnum {
    	public static void main(String[] args) {
    		Season spring = Season.SPRING;
    		System.out.println(spring);
    	}
    }
    enum Season{
    	SPRING,SUMMER,AUTUMN,WINTER
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    示例代码:

    public class TestEnum {
    	public static void main(String[] args) {
    		Season spring = Season.SPRING;
    		System.out.println(spring);
    	}
    }
    enum Season{
    	SPRING("春"),SUMMER("夏"),AUTUMN("秋"),WINTER("冬");
    	private final String description;
    	
    	private Season(String description){
    		this.description = description;
    	}
    	
    	public String toString(){//需要手动编写,无法使用Generate toString()...
    		return description;
    	}
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18

    枚举类的要求和特点:

    • 枚举类的常量对象列表必须在枚举类的首行,因为是常量,所以建议大写。
    • 如果常量对象列表后面没有其他代码,那么“;”可以省略,否则不可以省略“;”。
    • 编译器给枚举类默认提供的是private的无参构造,如果枚举类需要的是无参构造,就不需要声明,写常量对象列表时也不用加参数,
    • 如果枚举类需要的是有参构造,需要手动定义private的有参构造,调用有参构造的方法就是在常量对象名后面加(实参列表)就可以。
    • 枚举类默认继承的是java.lang.Enum类,因此不能再继承其他的类型
    • JDK1.5之后switch,提供支持枚举类型,case后面可以写枚举常量名。
    • 枚举类型如有其它属性,建议(不是必须)这些属性也声明为final的,因为常量对象在逻辑意义上应该不可变。

    枚举类型常用方法

    1.toString(): 默认返回的是常量名(对象名),可以继续手动重写该方法!
    2.name():返回的是常量名(对象名) 【很少使用】
    3.ordinal():返回常量的次序号,默认从0开始
    4.values():返回该枚举类的所有的常量对象,返回类型是当前枚举的数组类型,是一个静态方法
    5.valueOf(String name):根据枚举常量对象名称获取枚举对象
    
    • 1
    • 2
    • 3
    • 4
    • 5

    示例代码:

    public class TestEnum {
    	public static void main(String[] args) {
    		Season[] values = Season.values();
    		for (int i = 0; i < values.length; i++) {
    			switch(values[i]){
    			case SPRING:
    				System.out.println(values[i]+":春暖花开,万物复苏");
    				break;
    			case SUMMER:
    				System.out.println(values[i]+":百花争艳,郁郁葱葱");
    				break;
    			case AUTUMN:
    				System.out.println(values[i]+":菊桂飘香,百树凋零");
    				break;
    			case WINTER:
    				System.out.println(values[i]+":梅花独开,大地一色");
    				break;
    			}
    		}
    	}
    }
    enum Season{
    	SPRING,SUMMER,AUTUMN,WINTER
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24

    案例:
    1、声明月份枚举类Month:

    (1)创建:1-12月常量对象

    JANUARY,FEBRUARY,MARCH,APRIL,MAY,JUNE,JULY,AUGUST,SEPTEMBER,OCTOBER,NOVEMBER,DECEMBER
    
    • 1

    (2)声明两个属性:value(月份值,例如:JANUARY的value为1),
    description(描述,例如:JANUARY的description为1月份是一年的开始)。

    (3)声明一个有参构造,创建12个对象

    (4)声明一个方法:public static Month getByValue(int value)

    (5)手动重写toString():返回对象信息,例如:1->1月份是一年的开始。

    enum Month{
        JANUARY(1,"1月份是一年的开始"),FEBRUARY(2,"2月份是一年的开始"),MARCH(3,"3月份是一年的开始"),APRIL(4,"4月份是一年的开始");
    
        int value;
        String description;
    
        private Month(int value,String description){
            this.value=value;
            this.description=description;
        }
    
        public static Month getByValue(int value){
    
            for (Month month : Month.values()) {
                if(month.value==value){
                    return month;
                }
            }
    
            return null;
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22

    2、在测试类中,从键盘输入1个1-12的月份值,获取对应的月份对象,并打印对象

    package com.gec.demo2;
    
    import java.util.Scanner;
    
    /*
    * 枚举类
    * */
    enum Month{
    
        JANUARY(1,"1月份是一年的开始"),FEBRUARY(2,"2月份是一年的开始"),MARCH(3,"3月份是一年的开始"),APRIL(4,"4月份是一年的开始");
    
        int value;
        String description;
    
    
    
        private Month(int value,String description){
            this.value=value;
            this.description=description;
        }
    
        public static Month getByValue(int value){
    
            for (Month month : Month.values()) {
                if(month.value==value)
                    return month;
            }
            return null;
        }
    
        @Override
        public String toString() {
    
            return this.value+"->"+this.description;
        }
    }
    
    public class MonthMainTest {
    
        public static void main(String[] args) {
    
            int value=2;
            //从键盘输出
            Scanner scanner=new Scanner(System.in);
    
            System.out.println("请输入月份");
            int monthValue=scanner.nextInt();
    
            Month curMonth = Month.getByValue(monthValue);
            System.out.println(curMonth.toString());
    
        }
    }
    
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55

    包装类

    Java提供了两个类型系统,基本类型与引用类型,使用基本类型在于效率,然而当要使用只针对对象设计的API或新特性(例如泛型),那么基本数据类型的数据就需要用包装类来包装。

    序号基本数据类型包装类(java.lang包)
    1byteByte
    2shortShort
    3intInteger
    4longLong
    5floatFloat
    6doubleDouble
    7charCharacter
    8booleanBoolean
    9voidVoid

    装箱与拆箱

    装箱:把基本数据类型转为包装类对象。

    转为包装类的对象,是为了使用专门为对象设计的API和特性

    拆箱:把包装类对象拆为基本数据类型。

    转为基本数据类型,一般是因为需要运算,Java中的大多数运算符是为基本数据类型设计的。比较、算术等

    基本数值---->包装对象

    Integer i1 = new Integer(4);//使用构造函数函数
    Integer i2 = Integer.valueOf(4);//使用包装类中的valueOf方法
    
    
    • 1
    • 2
    • 3

    包装对象---->基本数值

    Integer i1 = new Integer(4);
    int num1 = i1.intValue();
    
    
    • 1
    • 2
    • 3

    JDK1.5之后,可以自动装箱与拆箱。

    注意:只能与自己对应的类型之间才能实现自动装箱与拆箱。

    Integer i = 4;//自动装箱。相当于Integer i = Integer.valueOf(4);
    i = i + 5;//等号右边:将i对象转成基本数值(自动拆箱) i.intValue() + 5;
    //加法运算完成后,再次装箱,把基本数值转成对象。
    
    
    • 1
    • 2
    • 3
    • 4
    Integer i = 1;
    Double d = 1;//错误的,1是int类型
    
    
    • 1
    • 2
    • 3

    总结:对象(引用数据类型)能用的运算符有哪些?

    (1)instanceof

    (2)=:赋值运算符

    (3)==和!=:用于比较地址,但是要求左右两边对象的类型一致或者是有父子类继承关系。

    (4)对于字符串这一种特殊的对象,支持“+”,表示拼接。

    包装类的一些API

    1、基本数据类型和字符串之间的转换

    (1)把基本数据类型转为字符串

    int a = 10;
    //String str = a;//错误的
    //方式一:
    String str = a + "";
    //方式二:
    String str = String.valueOf(a);
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    (2)把字符串转为基本数据类型

    String转换成对应的基本类型 ,除了Character类之外,其他所有包装类都具有parseXxx静态方法可以将字符串参数转换为对应的基本类型,例如:

    • public static int parseInt(String s):将字符串参数转换为对应的int基本类型。
    • public static long parseLong(String s):将字符串参数转换为对应的long基本类型。
    • public static double parseDouble(String s):将字符串参数转换为对应的double基本类型。

    或把字符串转为包装类,然后可以自动拆箱为基本数据类型

    • public static Integer valueOf(String s):将字符串参数转换为对应的Integer包装类,然后可以自动拆箱为int基本类型
    • public static Long valueOf(String s):将字符串参数转换为对应的Long包装类,然后可以自动拆箱为long基本类型
    • public static Double valueOf(String s):将字符串参数转换为对应的Double包装类,然后可以自动拆箱为double基本类型

    注意:如果字符串参数的内容无法正确转换为对应的基本类型,则会抛出java.lang.NumberFormatException异常。

    int a = Integer.parseInt("整数的字符串");
    double d = Double.parseDouble("小数的字符串");
    boolean b = Boolean.parseBoolean("true或false");
    
    int a = Integer.valueOf("整数的字符串");
    double d = Double.valueOf("小数的字符串");
    boolean b = Boolean.valueOf("true或false");
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

    2、数据类型的最大最小值

    Integer.MAX_VALUE和Integer.MIN_VALUE
    Long.MAX_VALUE和Long.MIN_VALUE
    Double.MAX_VALUE和Double.MIN_VALUE
    
    
    • 1
    • 2
    • 3
    • 4

    3、字符转大小写

    Character.toUpperCase('x');
    Character.toLowerCase('X');
    
    
    • 1
    • 2
    • 3

    4、整数转进制

    Integer.toBinaryString(int i) 
    Integer.toHexString(int i)
    Integer.toOctalString(int i)
    
    
    • 1
    • 2
    • 3
    • 4

    包装类对象的缓存问题

    包装类缓存对象
    Byte-128~127
    Short-128~127
    Integer-128~127
    Long-128~127
    Float没有
    Double没有
    Character0~127
    Booleantrue和false
    Integer i = 1;
    Integer j = 1;
    System.out.println(i == j);//true
    
    Integer i = 128;
    Integer j = 128;
    System.out.println(i == j);//false
    
    Integer i = new Integer(1);//新new的在堆中
    Integer j = 1;//这个用的是缓冲的常量对象,在方法区
    System.out.println(i == j);//false
    
    Integer i = new Integer(1);//新new的在堆中
    Integer j = new Integer(1);//另一个新new的在堆中
    System.out.println(i == j);//false
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    	@Test
    	public void test3(){
    		Double d1 = 1.0;
    		Double d2 = 1.0;
    		System.out.println(d1==d2);//false 比较地址,没有缓存对象,每一个都是新new的
    	}
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    1、类型转换问题

    	@Test
    	public void test4(){
    		Double d1 = 1.0;
    		double d2 = 1.0;
    		System.out.println(d1==d2);//true 和基本数据类型比较会自动拆箱,比较数据值
    	}
    	
    	@Test
    	public void test2(){
    		Integer i = 1000;
    		double j = 1000;
    		System.out.println(i==j);//true  会先将i自动拆箱为int,然后根据基本数据类型“自动类型转换”规则,转为double比较
    	}
    	
    	@Test
    	public void test(){
    		Integer i = 1000;
    		int j = 1000;
    		System.out.println(i==j);//true 会自动拆箱,按照基本数据类型进行比较
    	}
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21

    2、不可变对象

    public class TestExam {
    	public static void main(String[] args) {
    		int i = 1;
    		Integer j = new Integer(2);
    		Circle c = new Circle();
    		change(i,j,c);
    		System.out.println("i = " + i);//1
    		System.out.println("j = " + j);//2
    		System.out.println("c.radius = " + c.radius);//10.0
    	}
    	
    	/*
    	 * 方法的参数传递机制:
    	 * (1)基本数据类型:形参的修改完全不影响实参
    	 * (2)引用数据类型:通过形参修改对象的属性值,会影响实参的属性值
    	 * 这类Integer等包装类对象是“不可变”对象,即一旦修改,就是新对象,和实参就无关了
    	 */
    	public static void change(int a ,Integer b,Circle c ){
    		a += 10;
    //		b += 10;//等价于  b = new Integer(b+10);
    		c.radius += 10;
    		/*c = new Circle();
    		c.radius+=10;*/
    	}
    }
    class Circle{
    	double radius;
    }
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
  • 相关阅读:
    java冒泡排序
    OpenAI即将推出新一代AI模型DALL-E 3;用AI进行天然产物药物发现的综述
    半年总结-忙碌中的思考
    服务器的初始化
    QT+OSG/osgEarth编译之五十二:osgVolume+Qt编译(一套代码、一套框架,跨平台编译,版本:OSG-3.6.5工具库osgVolume)
    GoLong的学习之路,进阶,标准库之并发(context)补充并发三部曲,你真的明白context吗?
    在mmdet 3.0系列中使用visualizer进行目标检测真实框和检测框的可视化和绘制 并保存为图像
    conda清华源安装cuda12.1的pytorch
    第八章《Java高级语法》第11节:泛型
    巧用Prometheus来扩展kubernetes调度器
  • 原文地址:https://blog.csdn.net/yuqu1028/article/details/126080171