热门标签 | 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);}
}


推荐阅读
  • 本文介绍了解决java开源项目apache commons email简单使用报错的方法,包括使用正确的JAR包和正确的代码配置,以及相关参数的设置。详细介绍了如何使用apache commons email发送邮件。 ... [详细]
  • 开发笔记:实验7的文件读写操作
    本文介绍了使用C++的ofstream和ifstream类进行文件读写操作的方法,包括创建文件、写入文件和读取文件的过程。同时还介绍了如何判断文件是否成功打开和关闭文件的方法。通过本文的学习,读者可以了解如何在C++中进行文件读写操作。 ... [详细]
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • 纠正网上的错误:自定义一个类叫java.lang.System/String的方法
    本文纠正了网上关于自定义一个类叫java.lang.System/String的错误答案,并详细解释了为什么这种方法是错误的。作者指出,虽然双亲委托机制确实可以阻止自定义的System类被加载,但通过自定义一个特殊的类加载器,可以绕过双亲委托机制,达到自定义System类的目的。作者呼吁读者对网上的内容持怀疑态度,并带着问题来阅读文章。 ... [详细]
  • 本文介绍了Java中Currency类的getInstance()方法,该方法用于检索给定货币代码的该货币的实例。文章详细解释了方法的语法、参数、返回值和异常,并提供了一个示例程序来说明该方法的工作原理。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 本文主要解析了Open judge C16H问题中涉及到的Magical Balls的快速幂和逆元算法,并给出了问题的解析和解决方法。详细介绍了问题的背景和规则,并给出了相应的算法解析和实现步骤。通过本文的解析,读者可以更好地理解和解决Open judge C16H问题中的Magical Balls部分。 ... [详细]
  • JavaSE笔试题-接口、抽象类、多态等问题解答
    本文解答了JavaSE笔试题中关于接口、抽象类、多态等问题。包括Math类的取整数方法、接口是否可继承、抽象类是否可实现接口、抽象类是否可继承具体类、抽象类中是否可以有静态main方法等问题。同时介绍了面向对象的特征,以及Java中实现多态的机制。 ... [详细]
  • 本文介绍了Swing组件的用法,重点讲解了图标接口的定义和创建方法。图标接口用来将图标与各种组件相关联,可以是简单的绘画或使用磁盘上的GIF格式图像。文章详细介绍了图标接口的属性和绘制方法,并给出了一个菱形图标的实现示例。该示例可以配置图标的尺寸、颜色和填充状态。 ... [详细]
  • iOS超签签名服务器搭建及其优劣势
    本文介绍了搭建iOS超签签名服务器的原因和优势,包括不掉签、用户可以直接安装不需要信任、体验好等。同时也提到了超签的劣势,即一个证书只能安装100个,成本较高。文章还详细介绍了超签的实现原理,包括用户请求服务器安装mobileconfig文件、服务器调用苹果接口添加udid等步骤。最后,还提到了生成mobileconfig文件和导出AppleWorldwideDeveloperRelationsCertificationAuthority证书的方法。 ... [详细]
  • 本文介绍了在MFC下利用C++和MFC的特性动态创建窗口的方法,包括继承现有的MFC类并加以改造、插入工具栏和状态栏对象的声明等。同时还提到了窗口销毁的处理方法。本文详细介绍了实现方法并给出了相关注意事项。 ... [详细]
  • 解决Sharepoint 2013运行状况分析出现的“一个或多个服务器未响应”问题的方法
    本文介绍了解决Sharepoint 2013运行状况分析中出现的“一个或多个服务器未响应”问题的方法。对于有高要求的客户来说,系统检测问题的存在是不可接受的。文章详细描述了解决该问题的步骤,包括删除服务器、处理分布式缓存留下的记录以及使用代码等方法。同时还提供了相关关键词和错误提示信息,以帮助读者更好地理解和解决该问题。 ... [详细]
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社区 版权所有