热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

初学反射基本原理

反射:框架设计的灵魂*框架:半成品软件。可以在框架的基础上进行软件开发,简化编码*反射:将类的各个组成部分封装为其他对象
反射:框架设计的灵魂

* 框架:半成品软件。可以在框架的基础上进行软件开发,简化编码
* 反射:将类的各个组成部分封装为其他对象,这就是反射机制* 好处:1. 可以在程序运行过程中,操作这些对象。2. 可以解耦,提高程序的可扩展性。

在这里插入图片描述

Class类

* 获取Class类的方式:1. Class.forName("全类名"):将字节码文件加载进内存,返回Class类* 多用于配置文件,将类名定义在配置文件中。读取文件,加载类2. 类名.class:通过类名的属性class获取* 多用于参数的传递3. 对象.getClass():getClass()方法在Object类中定义着。* 多用于对象的获取字节码的方式* 结论:同一个字节码文件(*.class)在一次程序运行过程中,只会被加载一次,不论通过哪一种方式获取的Class对象都是同一个。

public class People {public void sleep(){System.out.println("睡觉");}
}
public class User {private String name;private int age;public String address;public User() {}public User(String name, int age, String address) {this.name = name;this.age = age;this.address = address;}public String getName() {System.out.println("haha");return name;}public String getName(String value) {System.out.println("haha"+value);return name;}public void setName(String name) {this.name = name;}public void eat(){System.out.println("吃饭饭");}
}public class ReflectClass {public static void main(String[] args) throws ClassNotFoundException {//获取Class类的 3 种方式Class<?> class1 &#61; Class.forName("reflect.entity.User");Class<User> class2 &#61; User.class;User user &#61; new User();Class<? extends User> class3 &#61; user.getClass();User user1 &#61; new User();Class<? extends User> class4 &#61; user1.getClass();System.out.println(class1);System.out.println(class2);System.out.println(class3);System.out.println(class4);//都是true 说明是同一份 *.class 文件System.out.println(class1&#61;&#61;class2);//trueSystem.out.println(class1&#61;&#61;class3);//trueSystem.out.println(class3&#61;&#61;class4);//true}
}

* Class对象功能&#xff1a;* 获取功能&#xff1a;1. 获取成员变量们* Field[] getFields() &#xff1a;获取所有public修饰的成员变量* Field getField(String name) 获取指定名称的 public修饰的成员变量* Field[] getDeclaredFields() 获取所有的成员变量&#xff0c;不考虑修饰符* Field getDeclaredField(String name) 2. 获取构造方法们* Constructor[] getConstructors() * Constructor getConstructor(类... parameterTypes) * Constructor getDeclaredConstructor(类... parameterTypes) * Constructor[] getDeclaredConstructors() 3. 获取成员方法们&#xff1a;* Method[] getMethods() * Method getMethod(String name, 类... parameterTypes) * Method[] getDeclaredMethods() * Method getDeclaredMethod(String name, 类... parameterTypes) 4. 获取全类名 * String getName()

Field类

* Field&#xff1a;成员变量* 操作&#xff1a;1. 设置值* void set(Object obj, Object value) 2. 获取值* get(Object obj) 3. 忽略访问权限修饰符的安全检查* setAccessible(true):暴力反射

public class ReflrctField {public static void main(String[] args) throws Exception {//0. 获取User 的 Class对象Class<User> userClass &#61; User.class;/*** getFields()&#xff1a;只获取所有 被public 修饰的成员变量* getField()&#xff1a;只获取指定名称的 被public 修饰的成员变量*///1.获取成员变量们&#xff08;public&#xff09; Field[] fields &#61; userClass.getFields();for (Field field : fields) {System.out.println(field);}//2、获取指定成员变量&#xff08;public&#xff09;Field address &#61; userClass.getField("address");User user &#61; new User();//get 返回所表示的字段的值,指定的对象上Object value &#61; address.get(user);System.out.println(value);//set 将指定对象参数上 此字段设置为指定的新值address.set(user,"广州");System.out.println(user);System.out.println("&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;");/*** getDeclaredFields()获取所有成员变量(不考虑修饰符) Declared&#xff1a;声明过的* getDeclaredField() 获取某个属性值( publick )*///3、getDeclaredFields()获取所有成员变量(不考虑修饰符) Declared&#xff1a;声明过的Field[] declaredFields &#61; userClass.getDeclaredFields();for (Field declaredField : declaredFields) {System.out.println(declaredField);}//4、getDeclaredField 获取某个属性值(publick)Field age &#61; userClass.getDeclaredField("age");//忽略访问权限修饰符安全检查age.setAccessible(true);//暴力反射System.out.println(age.get(user));}
}

Constructor类

* Constructor:构造方法* 创建对象&#xff1a;* T newInstance(Object... initargs) * 如果使用空参数构造方法创建对象&#xff0c;操作可以简化&#xff1a;Class对象的newInstance方法

public class ReflecConstructor {public static void main(String[] args) throws Exception {Class userClass &#61; User.class;//获取构造方法 getConstructor(),根据传入数据类型重载Constructor constructor &#61; userClass.getConstructor();System.out.println(constructor);//创建对象Object user &#61; constructor.newInstance();System.out.println(user);Constructor constructor1 &#61; userClass.getConstructor(String.class, int.class,String.class);System.out.println(constructor1);Object user1 &#61; constructor1.newInstance("hua", 19,"gua");System.out.println(user1);}
}

Method类

* Method&#xff1a;方法对象* 执行方法&#xff1a;* Object invoke(Object obj, Object... args) * 获取方法名称&#xff1a;* String getName:获取方法名

public class ReflectMethod {public static void main(String[] args) throws Exception {Class<User> userClass &#61; User.class;//获取指定名称方法Method nameMethod1 &#61; userClass.getMethod("getName");User user &#61; new User();//执行方法nameMethod1.invoke(user);//根据传入的参数类型 获取重载的方法Method nameMethod2 &#61; userClass.getMethod("getName", String.class);nameMethod2.invoke(user,"huahuahua");//获取所有 public 方法&#xff08;包括父类的&#xff09;Method[] methods &#61; userClass.getMethods();for (Method method : methods) {System.out.println(method);//method.setAccessible(true);}System.out.println("&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;&#61;");//获取所有 public 方法&#xff08;不包括父类的&#xff09;Method[] methods2 &#61; userClass.getDeclaredMethods();for (Method method : methods2) {System.out.println(method);}//获取类名String userClassName &#61; userClass.getName();System.out.println(userClassName);}
}

案例

* 案例&#xff1a;* 需求&#xff1a;写一个"框架"&#xff0c;不能改变该类的任何代码的前提下&#xff0c;可以帮我们创建任意类的对象&#xff0c;并且执行其中任意方法* 实现&#xff1a;1. 配置文件2. 反射* 步骤&#xff1a;1. 将需要创建的对象的全类名和需要执行的方法定义在配置文件中2. 在程序中加载读取配置文件3. 使用反射技术来加载类文件进内存4. 创建对象5. 执行方法在src目录下新建文件 pro.properties,内容如下
className&#61;reflect.entity.People
methodName&#61;sleep

//例子&#xff1a;不能改变类的任何代码&#xff0c;可以创建任意类对象&#xff0c;执行任意方法
public class ReflectTest {public static void main(String[] args) throws Exception {/* 以下操作 是用反射 替代这两个的其中一个User user &#61; new User();user.eat();People people &#61; new People();people.sleep();*///1、加载配置文件//创建Properties对象Properties properties &#61; new Properties();//通过类加载器&#xff08;可以加载src目录下任意文件&#xff09;加载ClassLoader classLoader &#61; ReflectTest.class.getClassLoader();InputStream is &#61; classLoader.getResourceAsStream("user.properties");properties.load(is);//2、获取配置文件中定义的数据String className &#61; properties.getProperty("className");String methodName &#61; properties.getProperty("methodName");//3、加载该类进内存Class<?> cls &#61; Class.forName(className);//4、创建对象Object obj &#61; cls.newInstance();//5、获取方法对象Method method &#61; cls.getMethod(methodName);//执行方法需指定对象method.invoke(obj);}
}


推荐阅读
  • andr ... [详细]
  • 本文详细探讨了JDBC(Java数据库连接)的内部机制,重点分析其作为服务提供者接口(SPI)框架的应用。通过类图和代码示例,展示了JDBC如何注册驱动程序、建立数据库连接以及执行SQL查询的过程。 ... [详细]
  • 从 .NET 转 Java 的自学之路:IO 流基础篇
    本文详细介绍了 Java 中的 IO 流,包括字节流和字符流的基本概念及其操作方式。探讨了如何处理不同类型的文件数据,并结合编码机制确保字符数据的正确读写。同时,文中还涵盖了装饰设计模式的应用,以及多种常见的 IO 操作实例。 ... [详细]
  • 本文介绍了如何通过 Maven 依赖引入 SQLiteJDBC 和 HikariCP 包,从而在 Java 应用中高效地连接和操作 SQLite 数据库。文章提供了详细的代码示例,并解释了每个步骤的实现细节。 ... [详细]
  • 本文详细介绍了Java中的访问器(getter)和修改器(setter),探讨了它们在保护数据完整性、增强代码可维护性方面的重要作用。通过具体示例,展示了如何正确使用这些方法来控制类属性的访问和更新。 ... [详细]
  • 本文介绍如何使用阿里云的fastjson库解析包含时间戳、IP地址和参数等信息的JSON格式文本,并进行数据处理和保存。 ... [详细]
  • 本文详细介绍了Java中org.w3c.dom.Text类的splitText()方法,通过多个代码示例展示了其实际应用。该方法用于将文本节点在指定位置拆分为两个节点,并保持在文档树中。 ... [详细]
  • 本文探讨了如何优化和正确配置Kafka Streams应用程序以确保准确的状态存储查询。通过调整配置参数和代码逻辑,可以有效解决数据不一致的问题。 ... [详细]
  • 本文详细介绍了 Apache Jena 库中的 Txn.executeWrite 方法,通过多个实际代码示例展示了其在不同场景下的应用,帮助开发者更好地理解和使用该方法。 ... [详细]
  • 2023年京东Android面试真题解析与经验分享
    本文由一位拥有6年Android开发经验的工程师撰写,详细解析了京东面试中常见的技术问题。涵盖引用传递、Handler机制、ListView优化、多线程控制及ANR处理等核心知识点。 ... [详细]
  • 文件描述符、文件句柄与打开文件之间的关联解析
    本文详细探讨了文件描述符、文件句柄和打开文件之间的关系,通过具体示例解释了它们在操作系统中的作用及其相互影响。 ... [详细]
  • 深入理解Java泛型:JDK 5的新特性
    本文详细介绍了Java泛型的概念及其在JDK 5中的应用,通过具体代码示例解释了泛型的引入、作用和优势。同时,探讨了泛型类、泛型方法和泛型接口的实现,并深入讲解了通配符的使用。 ... [详细]
  • 并发编程:深入理解设计原理与优化
    本文探讨了并发编程中的关键设计原则,特别是Java内存模型(JMM)的happens-before规则及其对多线程编程的影响。文章详细介绍了DCL双重检查锁定模式的问题及解决方案,并总结了不同处理器和内存模型之间的关系,旨在为程序员提供更深入的理解和最佳实践。 ... [详细]
  • Java编程实践:深入理解方法重载
    本文介绍了Java中方法重载的概念及其应用。通过多个示例,详细讲解了如何在同一类中定义具有相同名称但不同参数列表的方法,以实现更灵活的功能调用。 ... [详细]
  • 深入探讨CPU虚拟化与KVM内存管理
    本文详细介绍了现代服务器架构中的CPU虚拟化技术,包括SMP、NUMA和MPP三种多处理器结构,并深入探讨了KVM的内存虚拟化机制。通过对比不同架构的特点和应用场景,帮助读者理解如何选择最适合的架构以优化性能。 ... [详细]
author-avatar
手机用户2502886253
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有