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

简介: 【4月更文挑战第14天】枚举类型(enum)是固定常量集合,Java中用`enum`定义。特点包括:使用enum关键字,定义类型名和值,可独立或嵌入定义,可实现接口,定义变量和方法。枚举在switch语句中适用,每个枚举值在JVM中唯一,不能继承Enum类。

什么是枚举类

枚举类型(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 ,然后在代码中为用到的类创建对应的 ArrayList<类型>;,既实现了编写一次,万能匹配,又通过编译器保证了类型安全;

  • 使用泛型时,把泛型参数 替换为需要的class类型
  • 不指定泛型参数类型时,编译器会给出警告,且只能将 视为 Object 类型
  • 可以在接口中定义泛型类型,实现此接口的类必须实现正确的泛型类型

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

  • 静态方法不能引用泛型类型 ,必须定义其他类型(例如 )来实现静态泛型方法;
  • 泛型可以同时定义多种类型,例如 Map 。

类型擦除:

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

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

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

  • 泛型类型必须⽤限定内的类型来进⾏初始化,否则会导致编译错误。
  • ⾮限定通配符表⽰可以⽤任意泛型类型来替代,类型为

泛型中 K T V E ?的含义

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

List和原始类型 List 之间的区别

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

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

List<?>和 List之间的区别是什么?
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)

相关文章
|
6天前
|
安全 Java
Java多线程集合类
本文介绍了Java中线程安全的问题及解决方案。通过示例代码展示了使用`CopyOnWriteArrayList`、`CopyOnWriteArraySet`和`ConcurrentHashMap`来解决多线程环境下集合操作的线程安全问题。这些类通过不同的机制确保了线程安全,提高了并发性能。
|
11天前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
50 4
|
10天前
|
Java 大数据 API
14天Java基础学习——第1天:Java入门和环境搭建
本文介绍了Java的基础知识,包括Java的简介、历史和应用领域。详细讲解了如何安装JDK并配置环境变量,以及如何使用IntelliJ IDEA创建和运行Java项目。通过示例代码“HelloWorld.java”,展示了从编写到运行的全过程。适合初学者快速入门Java编程。
|
11天前
|
Java 编译器 开发者
Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面
本文探讨了Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面,帮助开发者提高代码质量和程序的健壮性。
27 2
|
14天前
|
Java 开发工具 Android开发
Kotlin教程笔记(26) -Kotlin 与 Java 共存(一)
Kotlin教程笔记(26) -Kotlin 与 Java 共存(一)
|
Java 数据库 容器
|
8天前
|
安全 Java 测试技术
Java并行流陷阱:为什么指定线程池可能是个坏主意
本文探讨了Java并行流的使用陷阱,尤其是指定线程池的问题。文章分析了并行流的设计思想,指出了指定线程池的弊端,并提供了使用CompletableFuture等替代方案。同时,介绍了Parallel Collector库在处理阻塞任务时的优势和特点。
|
17天前
|
安全 Java
java 中 i++ 到底是否线程安全?
本文通过实例探讨了 `i++` 在多线程环境下的线程安全性问题。首先,使用 100 个线程分别执行 10000 次 `i++` 操作,发现最终结果小于预期的 1000000,证明 `i++` 是线程不安全的。接着,介绍了两种解决方法:使用 `synchronized` 关键字加锁和使用 `AtomicInteger` 类。其中,`AtomicInteger` 通过 `CAS` 操作实现了高效的线程安全。最后,通过分析字节码和源码,解释了 `i++` 为何线程不安全以及 `AtomicInteger` 如何保证线程安全。
java 中 i++ 到底是否线程安全?
|
5天前
|
安全 Java 开发者
深入解读JAVA多线程:wait()、notify()、notifyAll()的奥秘
在Java多线程编程中,`wait()`、`notify()`和`notifyAll()`方法是实现线程间通信和同步的关键机制。这些方法定义在`java.lang.Object`类中,每个Java对象都可以作为线程间通信的媒介。本文将详细解析这三个方法的使用方法和最佳实践,帮助开发者更高效地进行多线程编程。 示例代码展示了如何在同步方法中使用这些方法,确保线程安全和高效的通信。
25 9
|
8天前
|
存储 安全 Java
Java多线程编程的艺术:从基础到实践####
本文深入探讨了Java多线程编程的核心概念、应用场景及其实现方式,旨在帮助开发者理解并掌握多线程编程的基本技能。文章首先概述了多线程的重要性和常见挑战,随后详细介绍了Java中创建和管理线程的两种主要方式:继承Thread类与实现Runnable接口。通过实例代码,本文展示了如何正确启动、运行及同步线程,以及如何处理线程间的通信与协作问题。最后,文章总结了多线程编程的最佳实践,为读者在实际项目中应用多线程技术提供了宝贵的参考。 ####