IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> Java基础强化(反射) -> 正文阅读

[Java知识库]Java基础强化(反射)

什么是反射

程序运行期间动态的创建一个类的实例对象,调用一个类中的成员,往类中添加成员,删除成员等。成员包括(属性、方法)

第一次听说这个概念,可能不理解,看一个案例,通过反射创建一个实例对象,并调用其中的一个方法:
现在有名叫AService、BService、CService三个类,这三个类中分别都有一个show()方法。接下来把其全类名放到一个字符串数组中,随机一个数字,根据下标取出对应的全类名,实例化一个这个类的对象,调用类里面的show()方法

public class AService {
    public void show(){
        System.out.println("AService中的show方法");
    }
}
public class BService {
    public void show(){
        System.out.println("BService中的show方法");
    }
}
public class CService {
    public void show(){
        System.out.println("CService中的show方法");
    }
}
private static void dynamic() throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException {
        String[] objects = new String[]{"com.example.service.AService", "com.example.service.BService", "com.example.service.CService"};
        int num = new Random().nextInt(3);
        String ref = objects[num];
        // 获取一个类的Class对象
        Class<?> clazz = Class.forName(ref);
        // 创建一个实例对象
        Object newInstance = clazz.newInstance();
        // 获取一个method对象
        Method showMethod = clazz.getMethod("show");
        // 调用一个对象中的方法
        showMethod.invoke(newInstance);
    }

在这里请问一下,创建实例对象的Object newInstance = clazz.newInstance();的这一行代码你能看出是创建哪个类的对象吗?
显然不知道,更何况我们而没有显示的创建了某个类的实例对象。创建实例对象的Object newInstance = clazz.newInstance();的这一行代码是在程序运行时才执行的。也就是说, 在程序运行的时候,才实例化了某个类的一个对象,然后又通过反射调用了这个实例对象中的方法
在没有使用反射的情况下,这个案例大致上以下这种类似的方法:

private static void nodynamic() {
        String[] objects = new String[]{"com.example.service.AService", "com.example.service.BService", "com.example.service.CService"};
        int num = new Random().nextInt(3);
        String ref = objects[num];
        switch (ref) {
            case "com.example.service.AService":
                AService aService = new AService();
                aService.show();
                break;
            case "com.example.service.BService":
                BService bService = new BService();
                bService.show();
                break;
            case "com.example.service.CService":
                CService cService = new CService();
                cService.show();
                break;
            default:
                System.out.println("没有找到对象");
                break;
        }
    }

不使用反射很明显就复杂了很多。通过这个案例先有一个简单的认识

接下来再通过一个案例加深以下对反射的理解。
在使用Spring的时候(注解方式),如果我们需要将某个类加载到Spring的容器当中,就会在类上面标注@Service、@Component等其它几个注解进行标记,然后指定注解扫描包,然后Spring就会扫描指定包下标记了这些注解的类,并将这些类加载到Spring的容器当中,模拟这个流程来做个案例演示:
步骤一、自定义一个注解@StudyService

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface StudyService {
    String value() default "";
}

步骤二、对我们的业务类,加上我们自定义的注解
业务类:

package com.example.service;

import com.example.anntion.StudyService;

@StudyService
public class AService {
    public void show(){
        System.out.println("AService中的show方法");
    }
}
package com.example.service;

public class BService {
    public void show(){
        System.out.println("BService中的show方法");
    }
}
package com.example.service;

import com.example.anntion.StudyService;

@StudyService
public class CService {
    public void show(){
        System.out.println("CService中的show方法");
    }
}

在这里插入图片描述
步骤三、我们在容器初始化的时候,我们需要扫描标有service注解的类,然后实例化后放进容器内。大概的代码如下:

// 存储扫描到的类
    List<String> clazzList = new ArrayList<>();
    // 存储标记有@StudyService注解的类
    Map<String, Object> ioc = new HashMap<>();

    public void scanClass() throws ClassNotFoundException, InstantiationException, IllegalAccessException {
        // 扫描com.example.service文件夹下的类
        System.out.println("===============开始扫描文件夹下的类文件===============");
        File classPath = new File(this.getClass().getClassLoader().getResource("com//example//service").getFile());
        for (File file : classPath.listFiles()) {
            clazzList.add("com.example.service." + file.getName().replace(".class", ""));
        }
        System.out.println("===============结束扫描文件夹下的类文件===============");
        System.out.println("===============开始扫描标记有@StudyService注解的类,并放到IOC容器当中===============");
        for (String data : clazzList) {
            Class<?> clazz = Class.forName(data);
            if (clazz.isAnnotationPresent(StudyService.class)) {
                Object newInstance = clazz.newInstance();
                ioc.put(clazz.getSimpleName(), newInstance);
            }
        }
        System.out.println("===============结束扫描标记有@StudyService注解的类===============");
        System.out.println("===============查看IOC容器中的bean===============");
        for (Map.Entry<String, Object> entry : ioc.entrySet()) {
            System.out.println("key=" + entry.getKey() + ";value=" + entry.getValue());
        }
    }

运行结果:
在这里插入图片描述

反射带来的好处

提高了程序的灵活性与扩展性

原理

在这里插入图片描述

在这里插入图片描述

反射的优缺点

优点

可以在程序运行期间动态的去操作类的成员,动态的创建对象、调用对象中的属性与方法,

缺点

  • 如果我们在业务逻辑的代码当中,使用反射进行对象的操作,会模糊程序代码的逻辑,增大了维护的困难
  • 反射会消耗CPU资源,影响性能,在使用反射操作的时候,里面会对成员做一些的检查,例如:方法权限,然后获取成员对象,然后再返回拷贝之后的对象

反射的具体应用

  • 利用反射进行反编译
  • 加载数据库驱动
  • Spring将加了@Componet、@Service注解的类加载到IOC容器当中
  • Spring中xml的形式配置bean的时候,需要制定类的全类名,这也是通过反射实现的

反射的使用

package com.example.entity;

public class Student {
    private final static Integer num = 1;

    private Integer id;
    public String name;
    public Integer age;

    public void show(String message){
        System.out.println(message);
    }
}
 /**
     * 获取类Class对象的方式
     *
     * @throws ClassNotFoundException
     */
    private static void getClassMethod() throws ClassNotFoundException {
        Class<?> method01Clazz = Class.forName("com.example.entity.Student");
        Student student = new Student();
        Class<? extends Student> method02Clazz = student.getClass();
        Class<Student> method03Clazz = Student.class;
        getClassMethod();
    }
/**
     * 反射简单的使用
     *
     * @throws ClassNotFoundException
     * @throws InstantiationException
     * @throws IllegalAccessException
     * @throws NoSuchMethodException
     * @throws InvocationTargetException
     */
    private static void extracted() throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException {
        Class<?> clazz = Class.forName("com.example.entity.Student");

        // 创建一个类对象的实例
        Student newInstance = (Student) clazz.newInstance();
        System.out.println(newInstance);

        // 打印类对象里面的属性
        Field[] fields = clazz.getDeclaredFields();
        for (Field field : fields) {
            String modifier = Modifier.toString(field.getModifiers());
            Class<?> typeClazz = field.getType();
            String type = typeClazz.getSimpleName();
            String name = field.getName();
            System.out.println(modifier + " " + type + " " + name);
        }

        // 调用一个类对象的方法
        Method method = clazz.getMethod("show", String.class);
        method.invoke(newInstance, "你好啊");
    }

本文主要是加深对反射的理解

  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2021-08-07 11:51:51  更:2021-08-07 11:52:47 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年5日历 -2024/5/10 20:47:27-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码