Java-数据结构-反射、枚举 |ू・ω・` )

目录

❄️一、反射:

1、定义:

2、用途:

3、反射相关的类:

4、Class类:

1、Class类中相关的方法:

5、Class反射实例:

1、获得Class对象的三种方式:

2、反射的使用:

6、反射的优点和缺点:

❄️二、枚举:

1、定义:

2、使用:

3、枚举的优点和缺点:

❄️总结:


❄️一、反射:

1、定义:

Java的反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性,既然能拿到那么,我们就可以修改部分类型信息。

这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制。


2、用途:

1:当存在私有的成员变量、方法或者只对系统开放的,这个时候可以利用反射来获取所需的成员变量、方法。

2:可以开发各种类型的框架。


3、反射相关的类:

|------------------|------------------------------|
| 类名 | 用途 |
| Class类 | 代表类的实体,在运行的Java程序中表示类和接口 |
| Field类 | 代表类的成员变量/类的属性 |
| Method类 | 代表类的方法 |
| Constructor类 | 代表类的构造方法 |

在Java中大多数对象在运行的时候都会有:运行时类型 和 编译时类型 这两种类型,比如:

Person p1 = new Student();这个呢在编译时类型为Person,运行时类型为Student


4、Class类:

Class的帮助手册

代表类的实体,在运行的Java应用程序中表示类和接口。

1、Class类中相关的方法:

常用获得类相关的方法:

|-------------------------------|-------------------------------|
| 方法 | 用途 |
| getClassLoader() | 获得类的加载器 |
| getDeclaredClasses() | 返回一个数组,数组中包含该类中所有类和接口类的对象 |
| forName(String className) | 根据类名返回类的对象 |
| newInstance() | 创建类的实例 |
| getName() | 获得类的完整路径名字 |


常用获得类中属性相关的方法:

|-----------------------------------|-----------------|
| 方法 | 用途 |
| getField(String name) | 获得某个公有的属性对象 |
| getFields() | 获得所有共有的属性对象 |
| getDeclaredField(String name) | 获得某个属性对象 |
| getDeclaredFields() | 获得所有属性对象 |

注意:这些方法的返回值是 Field相关


获得类中注解相关的方法:

|--------------------------------------------------|-------------------------|
| 方法 | 用途 |
| getAnnotation(Class annotationClass) | 返回该类中与参数类型匹配的公有注解对象 |
| getAnnotations() | 返回该类所有的公有注解对象 |
| getDeclaredAnnotation(Class annotationClass) | 返回该类中与参数类型匹配的所有注解对象 |
| getDeclaredAnnotations() | 返回该类所有的注解对象 |


获得类中构造器相关的方法:

|---------------------------------------------------------|-------------------------|
| 方法 | 用途 |
| getConstructor(Class..<?> parameterTypes) | 获得该类中与参数类型匹配的公有构造方法 |
| getConstructors() | 获得该类的所有公有构造方法 |
| getDeclaredConstructor(Class..<?> parameterTypes) | 获得该类中与参数类型匹配的构造方法 |
| getDeclaredConstructors() | 获得该类所有构造方法 |

注意:这些方法的返回值为 Constructor相关


获得类中方法相关的方法:

|-----------------------------------------------------------------|-----------------|
| 方法 | 用途 |
| getMethod(String name,Class...<?> parameterTypes) | 获得该类某个公有的方法 |
| getMethods() | 获得该类所有公有的方法 |
| getDeclaredMethod(String name,Class...<?> parameterTypes) | 获得该类某个方法 |
| getDeclaredMethods() | 获得该类所有方法 |

注意:以上方法的返回值为 Method相关


5、Class反射实例:

1、获得Class对象的三种方式:

第一种:。

使用 Class.forName("类的全路径名"),静态方法,这个的使用前提:已明确类的全路径名

第二种:

使用 .class 方法,仅适合在编译前就已经明确要操作的 Class

第三种:

使用类对象的 getClass() 方法


我们来演示一下:

这个就是这三种获得 Class 对象的方法的演示了。我们再来看看 反射的使用。

注意:这里的第一种方法,会抛出一种异常,我们要对其进行处理这个异常。


2、反射的使用:

我们常用的方法呢就是第一种方法来使用反射。

调用不带参数的构造方法:

**我们在上面的方法中已经介绍了一种方法 ------**newInstance(),是用来创建类的实例的。

这个呢同样也存在异常的,所以我们同样要处理异常。

我们调用的是------不带参数的构造方法,而且我们在使用 Student 没有去 new 这个对象,就可以调用这个类。


我们在上面已经知道了关于 不带参数的构造方法 如何通过反射来调用的。那么对于 带参数的构造方法 是如何通过反射类调用的?我们来看看:

调用带参数的构造方法并且是私有的:

**1、我们这里使用的呢是 ------****getDeclaredConstructor,**这个方法。

2、我们的 Student 的成员变量呢有 String 和 int 类型,所以我们需要传参,传两个形参。

**3、我们使用 Constructor 来接收,这里要注意的是:在上面的 Student 中 不带参数的构造方法呢是私有的所以我们要使用一个方法 ------**setAccessible(true) 来告诉编译器,我们可以使用私有的属性

4、同样会抛出异常,所以我们需要解决这个异常

实现:

我们可以看到,这次呢调用的就是 有参数的构造方法了。


我们在来看看关于 私有的属性,我们要如何才能使用反射来调用 私有的属性 呢?我们来看:

调用 私有的属性:

1、对于 属性 我们使用返回 Field 的这些方法来调用,这里调用 私有属性 我们是使用 ------ getDeclaredField(String name) 方法。

2、因为是 私有的属性,所以我们需要使用 ------ setAccessible(true) 来处理

3、先把我们拿到的 class 对象使用 ------ newInstance() 方法来创建 Student 对象

4、使我们返回的 Field 来调用 set 方法来修改 私有的属性。

5、不要忘记处理它们的异常。

实现:

这里的 name 这个私有属性进行了改变。


我们来看看如何才能调用 私有方法 :

1、对于 方法 我们使用返回 Method 的这些方法来调用,这里调用 私有方法 我们是使用 ------ getDeclaredMethod(String name,Class...<?> parameterTypes) 方法。这里第一个参数为方法名称。第二个参数为参数列表。

2、因为是 私有的方法,所以我们需要使用 ------ setAccessible(true) 来处理

3、先把我们拿到的 class 对象使用 ------ newInstance() 方法来创建 Student 对象

4、使我们返回的 Method****来调用 invoke 方法来修改 私有的属性。

5、不要忘记处理它们的异常。

实现:

这样就可以调用 私有方法 了。


6、反射的优点和缺点:

优点:

1、对于任意一个类,都能够知道这个类的所有属性和方法

对于任意一个对象,都能够调用它的任意一个方法

2、增加程序的灵活性和扩展性,提高了自适应能力

缺点:

1、使用反射会有效率问题。会导致程序效率降低。

2、反射技术绕过了源代码的技术,因而会带来维护问题。反射代码比相应的直接代码更复杂 。


❄️二、枚举:

1、定义:

我们要是想要将一组常数组织起来的话呢,我们可能常用的方法呢就是这样的:

java 复制代码
public static final int RED = 1;
public static final int GREEN = 2;

但是呢,这种方法呢有不好的地方,比如:我们正好有个数字 1 ,那么它有可能误会为是 RED,这个时候呢,我们就可以使用 枚举 来实现,这样就有了类型------枚举类型。

java 复制代码
public enum TestEnum {
    RED,
    BLACK,
    GREEN;
}

优点: 将常量组织起来同一进行管理。

应用场景:错误状态码,消息类型,颜色的划分,状态机等等。

本质:是 java.lang.Enum 的子类,尽管是自己写的 枚举类,即是没有继承 enum 类,但是呢默 认继承了这个 enum 类


2、使用:

1)、switch 语句:

我们直接来看看代码如何实现的:

java 复制代码
public enum TestEnum {
    RED,
    BLACK,
    GREEN;

    public static void main(String[] args) {
        TestEnum testEnum = TestEnum.GREEN;
        switch (testEnum) {
            case RED:
                System.out.println("red");
                break;
            case BLACK:
                System.out.println("black");
                break;
            case GREEN:
                System.out.println("green");
                break;
            default:
                break;
        }
    }
}

2)、常用方法 :

Enum 的常用方法:

|-----------------|----------------------|
| 方法名称 | 描述 |
| values() | 以数组形式返回枚举类型的所有成员 |
| ordinal() | 获得枚举成员的索引位置 |
| valueOf() | 将普通字符串转换为枚举实例 |
| compareTo() | 比较两个枚举成员在定义时的顺序 |

示例:

java 复制代码
public enum TestEnum {
    RED,
    BLACK,
    GREEN;

    public static void main(String[] args) {
        //以数组形式返回枚举类型的所有成员
        TestEnum[] testEnum = TestEnum.values();
        for (int i = 0; i < testEnum.length; i++) {
            //获得枚举成员的索引位置
            System.out.println(testEnum[i] + " " + testEnum[i].ordinal());
        }

        //将普通字符串转换为枚举实例
        TestEnum testEnum1 = TestEnum.valueOf("RED");
        System.out.println(testEnum1);

        //比较两个枚举成员在定义时的顺序
        System.out.println(BLACK.compareTo(RED));
    }
}

我们说过,枚举类型 是一个类,就需要有构造方法,那么我们来看看 枚举类型 的构造方法:

java 复制代码
public enum TestEnum {
    RED("red",1),
    BLACK("black",2),
    GREEN("green",3);

    public String color;
    public int ordinal;
    
    private TestEnum(String color,int ordinal) {
        this.color = color;
        this.ordinal = ordinal;
    }
}

这里要注意:枚举的构造方法默认为私有的。


3、枚举的优点和缺点:

优点:

1、枚举常量更简单安全

2、枚举具有内置方法

缺点:

1、不可继承,无法拓展


❄️总结:

OK,这次的博客的分享就到这里结束了,我们下一篇还是介绍这个章节,让我们尽情期待吧!!!拜拜~~~

相关推荐
数据小爬虫@11 分钟前
Java爬虫实战:深度解析Lazada商品详情
java·开发语言
咕德猫宁丶13 分钟前
探秘Xss:原理、类型与防范全解析
java·网络·xss
songroom14 分钟前
Rust: offset祼指针操作
开发语言·算法·rust
code04号17 分钟前
C++练习:图论的两种遍历方式
开发语言·c++·图论
煤泥做不到的!2 小时前
挑战一个月基本掌握C++(第十一天)进阶文件,异常处理,动态内存
开发语言·c++
F-2H2 小时前
C语言:指针4(常量指针和指针常量及动态内存分配)
java·linux·c语言·开发语言·前端·c++
苹果酱05672 小时前
「Mysql优化大师一」mysql服务性能剖析工具
java·vue.js·spring boot·mysql·课程设计
chenziang12 小时前
leetcode hot100 环形链表2
算法·leetcode·链表
_oP_i3 小时前
Pinpoint 是一个开源的分布式追踪系统
java·分布式·开源
mmsx3 小时前
android sqlite 数据库简单封装示例(java)
android·java·数据库