JAVA基础 反射技术学习笔记 2

简介: JAVA基础 反射技术学习笔记

4.3 通过类的字节文件对象获取类的成员变量

方法介绍

image.png

package cn.it.bz.Reflect;
public class User {
    private String name;
    public int age;
   ……
}
package cn.it.bz.Reflect;
import java.lang.reflect.Field;
public class GetField {
    public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException {
        //获取类的字节文件对象
         Class<?> userClass = Class.forName("cn.it.bz.Reflect.User");
         //获取被public修饰的成员变量
         Field[] fields = userClass.getFields();
         for (Field field : fields) {
            System.out.println(field);
         }
        System.out.println("-----------------");
         //获取类中全部的成员变量
        Field[] declaredFields = userClass.getDeclaredFields();
        for (Field declaredField : declaredFields) {
            System.out.println(declaredField);
        }
        System.out.println("-----------------");
        //根据成员变量的名称返回被public修饰的成员变量
        Field name = userClass.getField("age");
        System.out.println(name);
        System.out.println("---------------------");
        //根据成员变量的名称返回成员变量,忽略修饰符
        Field name1 = userClass.getDeclaredField("name");
        System.out.println(name1);
    }
}

42ccd62b0ad0403aa951e7528c08dd87.png

4.4 操作成员变量

package cn.it.bz.Reflect;
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
public class GetField2 {
    public static void main(String[] args) throws Exception {
        Class<User> userClass = User.class;
        //获取public修饰的叫age的成员变量
        Field age = userClass.getField("age");
        //向User的实例user的age赋值
        User user = new User();
        age.set(user,123);
        //向User的实例user1的age赋值
        Constructor<User> constructor = userClass.getConstructor(null);//获取无参数构造方法
        User user1 = constructor.newInstance();
        age.set(user1,18);
        //获取成员变量的值
        System.out.println("user:"+user.getAge());
        System.out.println("user1:"+user1.getAge());
        System.out.println("-------或者---------");
        System.out.println(age.get(user));
        System.out.println(age.get(user1));
    }
}

4.5 通过类的字节文件对象获取类的方法

image.png

package cn.it.bz.Reflect;
import java.lang.reflect.Method;
public class GetMethod {
    public static void main(String[] args) throws Exception {
        User user = new User();
        Class<? extends User> aClass = user.getClass();
        //获取全部的方法(包含当前User类的全部方法和其继承父类的公共方法)
        Method[] methods = aClass.getMethods();
        for (Method method: methods) {
            System.out.println(method);
        }
        System.out.println("----------------------");
        //根据方法名和参数获得指定方法
         Method setName = aClass.getMethod("setName", String.class);
         System.out.println(setName);
        System.out.println("----------------------");
        //获取私有方法
        Method getName = aClass.getDeclaredMethod("suibian");
        System.out.println(getName);
    }
}

4.6 调用方法

package cn.it.bz.Reflect;
import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
public class GetMethod2 {
    public static void main(String[] args) throws Exception {
         Class<User> userClass = User.class;
         //获取方法区(也是字节文件对象)中的方法
         Method setName = userClass.getMethod("setName", String.class);
         //获取无参构造
         Constructor<User> constructor = userClass.getConstructor(null);
         //实例化User类
         User user = constructor.newInstance();
        //执行setName方法,第一个参数表示是哪个实例的setName方法,第二个参数表示执行该方法需要的参数。
        setName.invoke(user, "小明");
        //获取成员变量的值
        Method getName = userClass.getMethod("getName");
        Object invoke = getName.invoke(user);//表示执行user实例的getName方法,返回值是Object实际上是String类型
        System.out.println(invoke);
    }
}

4.7 获取类的其他信息

package cn.it.bz.Reflect;
public class GetClassInfo {
    public static void main(String[] args) {
        Class<User> userClass = User.class;
        //获取类的类名
        String name = userClass.getName();
        System.out.println("类的名称:"+name);
        //获取该类的包名
        Package aPackage = userClass.getPackage();
        String packageName = aPackage.getName();
        System.out.println("该类所在的包名:"+packageName);
        //获取该类的父类(Java不支持多继承,只有一个父类)
        Class superclass = userClass.getSuperclass();
        String superName = superclass.getName();
        System.out.println("该类的父类:"+superName);
        //获取该类实现的接口(一个类可以实现多个接口)
        Class[] interfaces = userClass.getInterfaces();
        for (Class anInterface: interfaces) {
            System.out.println("该类实现的接口:"+anInterface.getName());
        }
    }
}

五、小案例

需求:根据给定的方法名顺序来决定方法的执行顺序。

向main方法中String数组中传递参数:

package cn.it.bz.Reflect;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
class Reflect{
    public void method1(){
        System.out.println("method1执行");
    }
    public void method2(){
        System.out.println("method2执行");
    }
    public void method3(){
        System.out.println("method3执行");
    }
}
public class ReflectDemo {
    public static void main(String[] args) throws InvocationTargetException, IllegalAccessException {
       Reflect reflect = new Reflect();
       if (args.length > 0) {
           //获取Reflect类的字节文件对象
           Class<Reflect> reflectClass = Reflect.class;
           // 获取Reflect的所有方法
           Method[] methods = reflectClass.getMethods();
           for (String arg : args) {
               for (int i = 0; i < methods.length; i++) {
                   if(arg.equalsIgnoreCase(methods[i].getName())){
                       methods[i].invoke(reflect); //没有参数就不给
                   }
               }
           }
       }else {
           reflect.method1();
           reflect.method2();
           reflect.method3();
       }
    }
}

六、反射机制的效率

由于Java反射是要解析字节码,将内存中的对象进行解析,包括了一些动态类型,而JVM无法对这些代码进行优化。因此,反射操作的效率要比那些非反射操作低得多!

反射机制的效率测试

package cn.it.bz.Reflect;
import java.lang.reflect.Method;
public class Test {
    public static void main(String[] args) throws Exception {
        //获取当前时间毫秒数
        long l = System.currentTimeMillis();
        //实例化对象
        User user  = new User();
        //获取类字节文件对象
        Class<? extends User> aClass = user.getClass();
        //获取setName方法
        Method setNameMethod = aClass.getMethod("setName", String.class);
        //执行1000次
        for (int i = 0; i < 1000000000 ; i++) {
                setNameMethod.invoke(user, "name");
        }
        long l1 = System.currentTimeMillis();
        System.out.println("反射耗时::"+(l1-l));
        System.out.println("----------------------");
        long l2 = System.currentTimeMillis();
        User user1 = new User();
        for (int i = 0; i <1000000000;i++ ) {
            user1.setName("哈哈哈");
       }
        long l3 = System.currentTimeMillis();
        System.out.println("非反射射耗时::"+(l3-l2));
    }
}

七、setAccessible方法

setAccessible是启用和禁用访问安全检查的开关。值为 true 则指示反射的对象在使用时应该取消 Java 语言访问检查。值为 false 则指示反射的对象应该实施 Java 语言访问检查;默认值为false。

由于JDK的安全检查耗时较多.所以通过setAccessible(true)的方式关闭安全检查就可以达到提升反射速度的目的。

7.1 默认情况下获取private修饰的成员变量

package cn.it.bz.Reflect;
import java.lang.reflect.Field;
public class Test2 {
    public static void main(String[] args) throws Exception {
        User user = new User();
        Class<? extends User> aClass = user.getClass();
        //获取private修饰的成员变量name
        Field name = aClass.getDeclaredField("name");
        //向成员变量赋值
        name.set(user, "张三");
        //返回user对象的name属性
        String o =(String) name.get(user);
        System.out.println(o);
    }
}

7.2  忽略JDK安全检查

package cn.it.bz.Reflect;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
public class Test2 {
    public static void main(String[] args) throws Exception {
        User user = new User();
        Class<? extends User> aClass = user.getClass();
        //获取private修饰的成员变量name
        Field name = aClass.getDeclaredField("name");
        //忽略安全检查
        name.setAccessible(true);
        //向成员变量赋值
        name.set(user, "张三");
        //返回user对象的name属性
        String o =(String) name.get(user);
        System.out.println(o);
        System.out.println("----------------------");
        Method suibianMethod = aClass.getDeclaredMethod("suibian");
        //忽略安全检查
        suibianMethod.setAccessible(true);
        //执行suibianM方法
        suibianMethod.invoke(user);
    }
}

4f68923b5fa349ba9cd63e728f617d47.png

相关文章
|
20天前
|
存储 监控 安全
单位网络监控软件:Java 技术驱动的高效网络监管体系构建
在数字化办公时代,构建基于Java技术的单位网络监控软件至关重要。该软件能精准监管单位网络活动,保障信息安全,提升工作效率。通过网络流量监测、访问控制及连接状态监控等模块,实现高效网络监管,确保网络稳定、安全、高效运行。
46 11
|
29天前
|
XML Java 编译器
Java注解的底层源码剖析与技术认识
Java注解(Annotation)是Java 5引入的一种新特性,它提供了一种在代码中添加元数据(Metadata)的方式。注解本身并不是代码的一部分,它们不会直接影响代码的执行,但可以在编译、类加载和运行时被读取和处理。注解为开发者提供了一种以非侵入性的方式为代码提供额外信息的手段,这些信息可以用于生成文档、编译时检查、运行时处理等。
63 7
|
11天前
|
移动开发 前端开发 Java
Java最新图形化界面开发技术——JavaFx教程(含UI控件用法介绍、属性绑定、事件监听、FXML)
JavaFX是Java的下一代图形用户界面工具包。JavaFX是一组图形和媒体API,我们可以用它们来创建和部署富客户端应用程序。 JavaFX允许开发人员快速构建丰富的跨平台应用程序,允许开发人员在单个编程接口中组合图形,动画和UI控件。本文详细介绍了JavaFx的常见用法,相信读完本教程你一定有所收获!
Java最新图形化界面开发技术——JavaFx教程(含UI控件用法介绍、属性绑定、事件监听、FXML)
|
10天前
|
Java 数据库连接 Spring
反射-----浅解析(Java)
在java中,我们可以通过反射机制,知道任何一个类的成员变量(成员属性)和成员方法,也可以堆任何一个对象,调用这个对象的任何属性和方法,更进一步我们还可以修改部分信息和。
|
30天前
|
JavaScript 安全 Java
java版药品不良反应智能监测系统源码,采用SpringBoot、Vue、MySQL技术开发
基于B/S架构,采用Java、SpringBoot、Vue、MySQL等技术自主研发的ADR智能监测系统,适用于三甲医院,支持二次开发。该系统能自动监测全院患者药物不良反应,通过移动端和PC端实时反馈,提升用药安全。系统涵盖规则管理、监测报告、系统管理三大模块,确保精准、高效地处理ADR事件。
|
2月前
|
监控 Java
Java基础——反射
本文介绍了Java反射机制的基本概念和使用方法,包括`Class`类的使用、动态加载类、获取方法和成员变量信息、方法反射操作、以及通过反射了解集合泛型的本质。同时,文章还探讨了动态代理的概念及其应用,通过实例展示了如何利用动态代理实现面向切面编程(AOP),例如为方法执行添加性能监控。
|
2月前
|
监控 前端开发 Java
【技术开发】接口管理平台要用什么技术栈?推荐:Java+Vue3+Docker+MySQL
该文档介绍了基于Java后端和Vue3前端构建的管理系统的技术栈及功能模块,涵盖管理后台的访问、登录、首页概览、API接口管理、接口权限设置、接口监控、计费管理、账号管理、应用管理、数据库配置、站点配置及管理员个人设置等内容,并提供了访问地址及操作指南。
|
2月前
|
JSON 前端开发 JavaScript
java-ajax技术详解!!!
本文介绍了Ajax技术及其工作原理,包括其核心XMLHttpRequest对象的属性和方法。Ajax通过异步通信技术,实现在不重新加载整个页面的情况下更新部分网页内容。文章还详细描述了使用原生JavaScript实现Ajax的基本步骤,以及利用jQuery简化Ajax操作的方法。最后,介绍了JSON作为轻量级数据交换格式在Ajax应用中的使用,包括Java中JSON与对象的相互转换。
59 1
|
2月前
|
SQL 监控 Java
技术前沿:Java连接池技术的最新发展与应用
本文探讨了Java连接池技术的最新发展与应用,包括高性能与低延迟、智能化管理和监控、扩展性与兼容性等方面。同时,结合最佳实践,介绍了如何选择合适的连接池库、合理配置参数、使用监控工具及优化数据库操作,为开发者提供了一份详尽的技术指南。
40 7
|
2月前
|
Java
Java的反射
Java的反射。
39 2