Java基础教程(14)-Java中的枚举类,泛型和注解

举报
厨师之乡 发表于 2024/04/23 08:02:07 2024/04/23
【摘要】 什么是枚举类枚举类型(enum type)是指由一组固定的常量组成合法的类型。Java 中由关键字enum 来定义一个枚举类型Java 定义枚举类型的语句有以下特点:使用关键字 enum ;类型名称,比如这里的 Season ;一串允许的值,枚举可以单独定义在一个文件中,也可以嵌在其它 Java 类中;枚举可以实现一个或多个接口(Interface);可以定义新的变量 ;可以定义新的方法;...

什么是枚举类

枚举类型(enum type)是指由一组固定的常量组成合法的类型。Java 中由关键字enum 来定义一个枚举类型

Java 定义枚举类型的语句有以下特点:

  1. 使用关键字 enum ;
  2. 类型名称,比如这里的 Season ;
  3. 一串允许的值,
  4. 枚举可以单独定义在一个文件中,也可以嵌在其它 Java 类中;
  5. 枚举可以实现一个或多个接口(Interface);
  6. 可以定义新的变量 ;
  7. 可以定义新的方法;
  8. 可以定义根据具体枚举值而相异的类

enum 就和 class 一样,只是一个关键字,他并不是一个类;
当我们使用 enmu 来定义一个枚举类型的时候,编译器会自动帮我们创建一个 final类型的类继承 Enum 类,所以枚举类型不能被继承

Java 枚举如何比较

java 枚举值比较用双等于号和 equals 方法没啥区别,两个随便用都是一样的效果。

因为 enum 类型的每个常量在JVM中只有一个唯一实例; 枚举 Enum 类的 equals 方法默认实现就是通过 == 来比较的;

枚举类可以应用在 switch 语句中。因为枚举类天生具有类型信息和有限个枚举常量,所以比 int 、 String 类型更适合用在 switch 语句中.

什么是泛型:

Java 泛型( generics) 是 JDK 5 中引⼊的⼀个新特性, 允许在定义类和接口的时候使⽤类型参数( type parameter) 。
声明的类型参数在使⽤时⽤具体的类型来替换

泛型最⼤的好处是可以提⾼代码的复⽤性。 以 List 接⼜为例,我们可以将 String、Integer 等类型放⼊List 中, 如不⽤泛型, 存放 String 类型要写⼀个 List 接口, 存放Integer 要写另外⼀个 List 接口, 泛型可以很好的解决这个问题;

泛型就是定义一种模板,例如 ArrayList<T> ,然后在代码中为用到的类创建对应的 ArrayList<类型>;,既实现了编写一次,万能匹配,又通过编译器保证了类型安全;

  • 使用泛型时,把泛型参数 <T> 替换为需要的class类型

  • 不指定泛型参数类型时,编译器会给出警告,且只能将 <T> 视为 Object 类型

  • 可以在接口中定义泛型类型,实现此接口的类必须实现正确的泛型类型

  • 编写泛型时,需要定义泛型类型 <T> ;

  • 静态方法不能引用泛型类型 <T> ,必须定义其他类型(例如 <K> )来实现静态泛型方法;

  • 泛型可以同时定义多种类型,例如 Map<K, V> 。

类型擦除:

类型擦除指的是通过类型参数合并,将泛型类型实例关联到同一份字节码上。编译器只为泛型类型生成一份字节码,并将其实例关联到这份字节码上。

类型擦除的主要过程如下:

  1. 将所有的泛型参数用其最左边界(最顶级的父类型)类型替换
  2. 移除所有的类型参数。
  • 1、虚拟机中没有泛型,只有普通类和普通方法,所有泛型类的类型参数在编译时都会被擦除,泛型类并没有自己独有的 Class 类对象。比如并不存在 List<String>.class 或是List<Integer>.class,而只有 List.class。
  • 2、创建泛型对象时请指明类型,让编译器尽早的做参数检查;
  • 3、不要忽略编译器的警告信息,那意味着潜在的ClassCastException 等着你。
  • 4、 泛型的类型参数不能用在 Java 异常处理的 catch 语句中。因为异常处理是由JVM 在运行时刻来进行的。由于类型信息被擦除,JVM 是无法区分两个异常类型MyException<String>和MyException<Integer>的。对于 JVM 来说,它们都是MyException 类型的。也就无法执行与异常对应的 catch 语句。

通配符

通配符分为限定通配符和非限定通配符

  • 限定通配符对类型进⾏限制, 泛型中有两种限定通配符:

表示类型的上界,格式为:<? extends T>,即类型必须为 T 类型或者 T 子类

<? extends T> 允许调用读方法 T get() 获取 T 的引用,但不允许调用写方法 set(T) 传入 T 的引用(传入 null 除外)

表示类型的下界,格式为:<? super T>,即类型必须为 T 类型或者 T 的父类

<? super T> 允许调用写方法 set(T) 传入 T 的引用,但不允许调用读方法 Tget() 获取 T 的引用(获取 Object 除外)。
  • 泛型类型必须⽤限定内的类型来进⾏初始化,否则会导致编译错误。
  • ⾮限定通配符表⽰可以⽤任意泛型类型来替代,类型为<T>

泛型中 K T V E ?的含义

E - Element (在集合中使用,因为集合中存放的是元素)
T - Type(Java 类)
K - Key(键)
V - Value(值)
N - Number(数值类型)
? - 表示不确定的 java 类型(无限制通配符类型)
S、U、V - 2nd、3rd、4th types

List<Object>和原始类型 List 之间的区别

原始类型 List 和带参数类型 List<Object>之间的主要区别是,在编译时编译器不会对原始类型进行类型安全检查,却会对带参数的类型进行检查。通过使用 Object 作为类型,可以告知编译器该方法可以接受任何类型的对象,比如String 或 Integer。

它们之间的第二点区别是,你可以把任何带参数的类型传递给原始类型 List,但却不能把 List<String>传递给接受 List<Object>的方法,因为会产生编译错误

** List<?>和 List<Object>之间的区别是什么?**

List<?> 是一个未知类型的 List,而 List<Object> 其实是任意类型的 List。你可以把List<String>, List<Integer>赋值给 List<?>,却不能把 List<String>赋值给 List<Object>

什么是注解(Annotation)?

注解是放在Java源码的类、方法、字段、参数前的一种特殊“注释”:注解可以被编译器打包进入class文件,因此,注解是一种用作标注的“元数据”。

Java的注解可以分为三类:

  • 第一类是由编译器使用的注解,例如:

    @Override :让编译器检查该方法是否正确地实现了覆写;
    @SuppressWarnings :告诉编译器忽略此处代码产生的警告。
    这类注解不会被编译进入 .class 文件,它们在编译后就被编译器扔掉了。

  • 第二类是由工具处理 .class 文件使用的注解,比如有些工具会在加载class的时候,对class做动态修改,实现一些特殊的功能。这类注解会被编译进入 .class 文件,但加载结束后并不会存在于内存中。这类注解只被一些底层库使用,一般我们不必自己处理。

  • 第三类是在程序运行期能够读取的注解,它们在加载后一直存在于JVM中,这也是最常用的注解。

定义一个注解时,还可以定义配置参数。配置参数可以包括:

所有基本类型;
String;
枚举类型;
基本类型、String以及枚举的数组。

因为配置参数必须是常量,所以,上述限制保证了注解在定义时就已经确定了每个参数的值。

自定义注解

Java语言使用 @interface 语法来定义注解( Annotation )

package com.demo;

//定义一个注解
public @interface Report {
    int type() default 0;  //无参方法 有默认值
    String level() default "0";
    String value() default "";
}

注解的参数类似无参数方法,可以用 default 设定一个默认值(强烈推荐)。最常用的参数应当命名为 value 。

有一些注解可以修饰其他注解,这些注解就称为元注解(meta annotation)。Java标准库已经定义了一些元注解,我们只需要使用元注解,通常不需要自己去编写元注解。

package com.demo;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

/**
 * 最常用的元注解是 @Target 。使用 @Target 可以定义 Annotation 能够被应用于源码的哪些位
 * 置:
 * 类或接口: ElementType.TYPE ;
 * 字段: ElementType.FIELD ;
 * 方法: ElementType.METHOD ;
 * 构造方法: ElementType.CONSTRUCTOR ;
 * 方法参数: ElementType.PARAMETER 。
 * 
 * 另一个重要的元注解 @Retention 定义了 Annotation 的生命周期:
 * 仅编译期: RetentionPolicy.SOURCE ;
 * 仅class文件: RetentionPolicy.CLASS ;
 * 运行期: RetentionPolicy.RUNTIME 。
 * 如果 @Retention 不存在,则该 Annotation 默认为 CLASS 。因为通常我们自定义
 * 的 Annotation 都是 RUNTIME ,所以,务必要加
 * 上 @Retention(RetentionPolicy.RUNTIME) 这个元注解
 * */
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Report {
    int type() default 0;  //无参方法 有默认值
    String level() default "0";
    String value() default "";
}

@Target 定义的 value 是 ElementType[] 数组,只有一个元素时,可以省略数组的写法。

定义 Annotation 的步骤:

  • 第一步,用 @interface 定义注解
  • 第二步,添加参数、默认值:
  • 第三步,用元注解配置注解:

读取自定义注解

注解定义后也是一种 class ,所有的注解都继承自java.lang.annotation.Annotation ,因此,读取注解,需要使用反射API;

Java提供的使用反射API读取 Annotation 的方法包括:

判断某个注解是否存在于 Class 、 Field 、 Method 或 Constructor :

Class.isAnnotationPresent(Class)
Field.isAnnotationPresent(Class)
Method.isAnnotationPresent(Class)
Constructor.isAnnotationPresent(Class)

使用反射API读取Annotation:

Class.getAnnotation(Class)
Field.getAnnotation(Class)
Method.getAnnotation(Class)
Constructor.getAnnotation(Class)

【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。