热门标签 | 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序列化对象传给PHP的方法及原理解析
    本文介绍了Java序列化对象传给PHP的方法及原理,包括Java对象传递的方式、序列化的方式、PHP中的序列化用法介绍、Java是否能反序列化PHP的数据、Java序列化的原理以及解决Java序列化中的问题。同时还解释了序列化的概念和作用,以及代码执行序列化所需要的权限。最后指出,序列化会将对象实例的所有字段都进行序列化,使得数据能够被表示为实例的序列化数据,但只有能够解释该格式的代码才能够确定数据的内容。 ... [详细]
  • 使用Ubuntu中的Python获取浏览器历史记录原文: ... [详细]
  • 本文介绍了Perl的测试框架Test::Base,它是一个数据驱动的测试框架,可以自动进行单元测试,省去手工编写测试程序的麻烦。与Test::More完全兼容,使用方法简单。以plural函数为例,展示了Test::Base的使用方法。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • Java在运行已编译完成的类时,是通过java虚拟机来装载和执行的,java虚拟机通过操作系统命令JAVA_HOMEbinjava–option来启 ... [详细]
  • Oracle seg,V$TEMPSEG_USAGE与Oracle排序的关系及使用方法
    本文介绍了Oracle seg,V$TEMPSEG_USAGE与Oracle排序之间的关系,V$TEMPSEG_USAGE是V_$SORT_USAGE的同义词,通过查询dba_objects和dba_synonyms视图可以了解到它们的详细信息。同时,还探讨了V$TEMPSEG_USAGE的使用方法。 ... [详细]
  • 本文介绍了Python爬虫技术基础篇面向对象高级编程(中)中的多重继承概念。通过继承,子类可以扩展父类的功能。文章以动物类层次的设计为例,讨论了按照不同分类方式设计类层次的复杂性和多重继承的优势。最后给出了哺乳动物和鸟类的设计示例,以及能跑、能飞、宠物类和非宠物类的增加对类数量的影响。 ... [详细]
  • Imtryingtofigureoutawaytogeneratetorrentfilesfromabucket,usingtheAWSSDKforGo.我正 ... [详细]
  • 本文介绍了在iOS开发中使用UITextField实现字符限制的方法,包括利用代理方法和使用BNTextField-Limit库的实现策略。通过这些方法,开发者可以方便地限制UITextField的字符个数和输入规则。 ... [详细]
  • 纠正网上的错误:自定义一个类叫java.lang.System/String的方法
    本文纠正了网上关于自定义一个类叫java.lang.System/String的错误答案,并详细解释了为什么这种方法是错误的。作者指出,虽然双亲委托机制确实可以阻止自定义的System类被加载,但通过自定义一个特殊的类加载器,可以绕过双亲委托机制,达到自定义System类的目的。作者呼吁读者对网上的内容持怀疑态度,并带着问题来阅读文章。 ... [详细]
  • 本文介绍了RxJava在Android开发中的广泛应用以及其在事件总线(Event Bus)实现中的使用方法。RxJava是一种基于观察者模式的异步java库,可以提高开发效率、降低维护成本。通过RxJava,开发者可以实现事件的异步处理和链式操作。对于已经具备RxJava基础的开发者来说,本文将详细介绍如何利用RxJava实现事件总线,并提供了使用建议。 ... [详细]
  • 本文讨论了微软的STL容器类是否线程安全。根据MSDN的回答,STL容器类包括vector、deque、list、queue、stack、priority_queue、valarray、map、hash_map、multimap、hash_multimap、set、hash_set、multiset、hash_multiset、basic_string和bitset。对于单个对象来说,多个线程同时读取是安全的。但如果一个线程正在写入一个对象,那么所有的读写操作都需要进行同步。 ... [详细]
  • Python语法上的区别及注意事项
    本文介绍了Python2x和Python3x在语法上的区别,包括print语句的变化、除法运算结果的不同、raw_input函数的替代、class写法的变化等。同时还介绍了Python脚本的解释程序的指定方法,以及在不同版本的Python中如何执行脚本。对于想要学习Python的人来说,本文提供了一些注意事项和技巧。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
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社区 版权所有