Java基础——自学习使用(反射)

一、反射的定义

        反射是java基础的核心,反射是获取类信息的能力。类信息包括:方法、变量、接口、父类、构造器、类名等。

Java的反射(Reflection)是一种强大的机制,它允许程序在运行时检查或修改类、接口、字段以及方法的行为。反射通常被用于以下几种情况:

  1. 动态地创建类的实例:通过反射,可以在运行时创建类的实例,而无需在编译时知道类的具体类型。

  2. 调用方法:可以通过反射调用类的任何方法,包括私有方法(尽管通常不推荐这样做,因为它破坏了封装性)。

  3. 访问和修改字段:可以访问和修改类的私有字段,这同样需要谨慎使用以避免违反封装原则。

  4. 检查类的属性和方法:反射可以用来在运行时获取类的所有方法、字段、构造函数等的信息。

  5. 实现泛型数组操作:由于Java的泛型擦除机制,反射可以用来处理泛型数组。

二、反射的作用 

        需要获取类信息,所以需要用到反射(new对象也能获取类信息但获取的类信息的能力有限)。

1. 获取Class对象

反射的起点是获取Class对象,它代表了正在运行的Java应用程序中的类和接口。

  • 使用.class语法:Class<?> clazz = String.class;
  • 使用Class.forName()方法:Class<?> clazz = Class.forName("java.lang.String"); 注意,这种方法会抛出ClassNotFoundException
  • 使用对象的getClass()方法:String str = "Hello"; Class<?> clazz = str.getClass();
2. 创建实例

通过Class对象,可以使用newInstance()方法(已过时,推荐使用getDeclaredConstructor()方法)或getDeclaredConstructor()Constructor.newInstance()组合来创建类的实例。

  • 访问方法:使用getMethod()getDeclaredMethod()获取Method对象,然后调用invoke()方法来执行方法。
  • 访问字段:使用getField()getDeclaredField()获取Field对象,然后调用get()set()方法来访问或修改字段的值。注意,私有字段在访问前可能需要调用setAccessible(true)

三、反射的工作原理

想要反射获取类信息就必须进入类对象阶段,有三种方式进入类对象阶段 。

①Class.forName("全类名"),全类名为包名加类名

②类名.class

③对象名.getclass()

四、反射的特点

        创建一个student类,创建一个test类。

//Student类
public class Student {private String name="张三";public Integer age=18;Character sex='男';protected Double height =185.5;public String color ="蓝色";public void run(){System.out.println("这是run()");}private int getAge(int age){System.out.println("age()");return age;}void  aaa(String name,Integer height){System.out.println("aaa()");}protected void haha(String name,int age){System.out.println("haha()");}public Student(){}private Student(String name){}public Student(String name,Integer age){}public Student(String name, Integer age, Character sex, Double height, String color) {this.name = name;this.age = age;this.sex = sex;this.height = height;this.color = color;}
}

1.想要获取类信息就必须进入类对象阶段,以下是进入类对象阶段的三种方式。

public class Test {public static void main(String[] args) throws Exception {//1、想要获取类信息就必须进入类对象阶段,以下是进入类对象阶段的三种方式Class class1=Class.forName("反射.Student");Class class2=Student.class;Student student =new Student();Class class3=student.getClass();System.out.println(class1==class2);System.out.println(class2==class3);System.out.println("------------");

2.以class1为例进行变量信息的获取。

 //2、以class1为例进行变量信息的获取//①获取全部的变量信息,并且打印Field[] files=class1.getDeclaredFields();System.out.println(Arrays.toString(files));System.out.println("------------");//②获取public修饰的全局变量信息,并打印Field[] fiels1=class1.getFields();System.out.println(Arrays.toString(fiels1));System.out.println("------------");//③获取任意一个变量信息Field nameField =class1.getDeclaredField("name");System.out.println(nameField);Field hightField =class1.getDeclaredField("height");System.out.println(hightField);Field ageField =class1.getDeclaredField("age");System.out.println(ageField);System.out.println("------------");//④获取任意一个public修饰的变量信息Field ageField1 =class1.getField("age");System.out.println(ageField1);System.out.println("------------");

3.以class1为例进行方法信息的获取。

 //3以class1为例进行方法信息的获取//①获取全部方法信息Method[] declaredMethods=class1.getDeclaredMethods();System.out.println(Arrays.toString(declaredMethods));System.out.println("------------");//②获取public修饰的方法信息Method[] methods =class1.getMethods();System.out.println(Arrays.toString(methods));System.out.println("------------");//③获取任意一个方法信息Method getAge =class1.getDeclaredMethod("getAge", int.class);Method aaa =class1.getDeclaredMethod("aaa", String.class, Integer.class);Method run=class1.getDeclaredMethod("run");Method hhh=class1.getDeclaredMethod("haha", String.class, int.class);System.out.println(hhh);System.out.println(aaa);System.out.println(run);System.out.println(getAge);System.out.println("------------");//④获取任意一个public修饰的方法信息Method run1 =class1.getMethod("run");System.out.println(run1);System.out.println("------------");

4.以class1为例进行构造器的获取,构造器一般都是public类型的若有其他类型用getDeclaredConstructors获取。

//4、以class1为例进行构造器的获取,构造器一般都是public类型的若有其他类型用getDeclaredConstructors获取
//①获取所有的构造器
Constructor[] declaredconstructors =class1.getDeclaredConstructors();
System.out.println(Arrays.toString(declaredconstructors));
System.out.println("------------");
//②获取public修饰的构造器
Constructor[] constructors =class1.getConstructors();
System.out.println(Arrays.toString(constructors));
System.out.println("------------");
//③获取任意某个public修饰的构造器
Constructor declaredconstructor =class1.getDeclaredConstructor();
Constructor declaredconstructor1 =class1.getDeclaredConstructor(String.class ,Integer.class);
Constructor declaredconstructor2 =class1.getDeclaredConstructor(String.class,Integer.class ,Character.class ,Double.class ,String.class);
System.out.println(declaredconstructor);
System.out.println(declaredconstructor1);
System.out.println(declaredconstructor2);
System.out.println("------------");

5.对构造器的调用

使用构造器创建对象

//5对构造器的调用
// ①public修饰的构造器的使用(以该代码所获取的构造器信息为例 Constructor declaredconstructor1 =class1.getDeclaredConstructor(String.class ,Integer.class))
//创建对象 需要强转
Student student1 = (Student) declaredconstructor1.newInstance("aaa",18);
System.out.println("------------");
// ②private修饰的构造器的调用,但想要使用必须暴力反射,而protected、和不写的不需要暴力反射
Constructor  declaredconstructor4=class1.getDeclaredConstructor(String.class);
declaredconstructor4.setAccessible(true);
Student student2= (Student) declaredconstructor4.newInstance("AAA");
System.out.println("------------");

6.对方法的调用

// 6对方法的调用(以该代码所获取的构造器信息为例 Constructor declaredconstructor1 =class1.getDeclaredConstructor(String.class ,Integer.class))
//创建该对象  Student student1 = (Student) declaredconstructor1.newInstance("aaa",18);
//①对public修饰方法的调用
Method run2= class1.getDeclaredMethod("run");
run2.invoke(student1);
System.out.println("------------");
//②对private修饰方法的调用 需要暴力反射
Method getAge1= class1.getDeclaredMethod("getAge", int.class);
getAge1.setAccessible(true);
getAge1.invoke(student1,18);
System.out.println("------------");
//③不写修饰的方法的调用
Method aaa1 =class1.getDeclaredMethod("aaa", String.class, Integer.class);
aaa1.invoke(student1,"admin",199);
System.out.println("------------");
//对protected修饰方法的调用
Method haha1=class1.getDeclaredMethod("haha", String.class, int.class);
haha1.invoke(student1,"admin",99);
System.out.println("------------");

7.对变量的调用赋值取值操作

// 7对变量的调用赋值取值操作(以该代码所获取的构造器信息为例 Constructor declaredconstructor1 =class1.getDeclaredConstructor(String.class ,Integer.class))//创建该对象  Student student1 = (Student) declaredconstructor1.newInstance("aaa",18);//①给privat修饰的变量赋值,取值操作,需要暴力反射Field name=class1.getDeclaredField("name");name.setAccessible(true);name.set(student1,"admin");System.out.println(name.get(student1));System.out.println("------------");//②其他类型不需要Field age=class1.getDeclaredField("age");age.set(student1,88);System.out.println(age.get(student1));System.out.println("------------");Field sex=class1.getDeclaredField("sex");sex.set(student1,'女');System.out.println(sex.get(student1));System.out.println("------------");Field height=class1.getDeclaredField("height");height.set(student1,188.88);System.out.println(height.get(student1));}
}

五、习题

        获取该类的构造方法来构造对象,获取该类的名称和该类中每一个变量的名称和方法名称,并执行相关方法,给没一个变量赋值,并获取值,执行每一个构造方法、执行相应的方法。

//获取该类的构造方法来构造对象,获取该类的名称和该类中每一个变量的名称和方法名称,并执行相关方法,给没一个变量赋值,并获取值,执行每一个构造方法、执行相应的方法
public class Student {private String name;private int age;private String address;public Student(String name, int age, String address) {this.name = name;this.age = age;this.address = address;}public String getName(){return name;}public void setName(String name){this.name=name;}public int getAge(){return age;}public void setAge(int age){this.age=age;}public String getAddress(){return address;}public void  setAddress(String address){this.address=address;}
}
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;public class Test {public static void main(String[] args) throws Exception {Class class1=Class.forName("反射练习.Student");// 获取并打印类名System.out.println("类的名称: " + class1.getName());// 获取并打印所有字段(变量)名称System.out.println("变量名称为");Field[] fields = class1.getDeclaredFields();for (Field field : fields) {System.out.println( field.getName());}System.out.println("-----------");// 获取并打印所有方法名称(包括继承的方法)Method[] methods = class1.getMethods();System.out.println("方法名称为");for (Method method : methods) {System.out.println(  method.getName());}System.out.println("-----------");Constructor declaredconstructor1 =class1.getDeclaredConstructor(String.class ,int.class,String.class);Student student1= (Student) declaredconstructor1.newInstance("李四",18,"北京 ");Field name=class1.getDeclaredField("name");name.setAccessible(true);System.out.println(name.get(student1));Field age=class1.getDeclaredField("age");age.setAccessible(true);System.out.println(age.get(student1));Field address=class1.getDeclaredField("address");address.setAccessible(true);System.out.println(address.get(student1));System.out.println("-----------");Method setname= class1.getDeclaredMethod("setName", String.class);setname.invoke(student1,"张三");Method getname= class1.getDeclaredMethod("getName");getname.invoke(student1);Method setAge= class1.getDeclaredMethod("setAge", int.class);setAge.invoke(student1,20);Method getAge= class1.getDeclaredMethod("getAge");getAge.invoke(student1);Method setaddress= class1.getDeclaredMethod("setAddress", String.class);setaddress.invoke(student1,"上海");Method getaddress= class1.getDeclaredMethod("getAddress");getaddress.invoke(student1);Field name1=class1.getDeclaredField("name");name1.setAccessible(true);System.out.println(name1.get(student1));Field age1=class1.getDeclaredField("age");age1.setAccessible(true);System.out.println(age1.get(student1));Field address1=class1.getDeclaredField("address");address1.setAccessible(true);System.out.println(address1.get(student1));}
}

输出结果

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.xdnf.cn/news/1522478.html

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

【书生大模型实战营】MindSearch CPU-only 版部署

MindSearch CPU-only 版部署 MindSearch CPU-only 版部署任务步骤 MindSearch CPU-only 版部署 任务 将 MindSearch 部署到 HuggingFace 并美化 Gradio 的界面&#xff0c;并提供截图和 Hugging Face 的Space的链接。 步骤 按照官方教程&#xff0c;实现在网页上打开MindSe…

Java中等题-盛最多水的容器(力扣)

给定一个长度为 n 的整数数组 height 。有 n 条垂线&#xff0c;第 i 条线的两个端点是 (i, 0) 和 (i, height[i]) 。 找出其中的两条线&#xff0c;使得它们与 x 轴共同构成的容器可以容纳最多的水。 返回容器可以储存的最大水量。 说明&#xff1a;你不能倾斜容器。 我一开…

继电器实现上下电

一、依赖库 import serial 二、代码 import serialdef batonoff(com,status):继电器开关:param status::return:ser serial.Serial(com, baudrate9600, timeout0.1)if ser.is_open:if status "on":ser.write(data[0xA0, 0x01, 0x01, 0xA2])print(打开继电器)els…

Nginx负载均衡数据流分析

1、各机器ip信息 客户端IP&#xff1a;192.168.3.239 Nginx代理服务器IP&#xff1a;192.168.3.241 服务端IP&#xff1a;192.168.3.238 2、架构图&#xff08;略&#xff09; 3、 下图是在服务端上面的抓包分析。 下图是在客户端上面的抓包分析&#xff1a; 下图是在代理服务…

Python基础笔记

一、python基础1.1 基础知识1.1.1 注释 注释&#xff1a;在程序中对程序代码进行解释说明的文字。 作用&#xff1a;注释不是程序&#xff0c;不能被执行&#xff0c;只是对程序代码进行解释说明&#xff0c;让别人可以看懂程序代码的作用&#xff0c;能够大大增强程序的可读性…

【数论 状态机dp】2572. 无平方子集计数

本文涉及知识点 C动态规划 数论 质数、最大公约数、菲蜀定理 LeetCode 2572. 无平方子集计数 给你一个正整数数组 nums 。 如果数组 nums 的子集中的元素乘积是一个 无平方因子数 &#xff0c;则认为该子集是一个 无平方 子集。 无平方因子数 是无法被除 1 之外任何平方数整…

数据结构,单向链表

数据结构是计算机科学中的一个核心概念&#xff0c;它研究的是数据的组织、管理和存储方式&#xff0c;以及在这些数据上进行操作时的算法。数据结构为数据的高效访问和修改提供了基础。 数组&#xff08;Array&#xff09;&#xff1a;一种线性数据结构&#xff0c;可以存储固…

开源 AI 智能名片 O2O 商城小程序:引入淘汰机制,激发社交电商新活力

摘要&#xff1a;本文深入探讨在社交电商领域中&#xff0c;开源 AI 智能名片 O2O 商城小程序如何通过设置淘汰机制&#xff0c;实现“良币驱逐劣币”&#xff0c;激励士气&#xff0c;为社交电商企业注入新的活力。通过分析缺乏淘汰机制的弊端以及设置淘汰机制的优势&#xff…

一篇入门C语言【文件】

本科期间C语言的课本无论哪个版本都会有【文件】这一章节&#xff0c;不过好多学校基本上不讲或者就出一道选择题&#xff0c;讲得很浅&#xff0c;今天这篇详细总结一下这部分的知识~ 一.原理解析 文件是指存在在外部介质&#xff08;如磁盘、磁带&#xff09;上的数据集合。操…

NPDP|如何在传统行业中做好产品管理的策略与建议

在当今这个快速变化的数字时代&#xff0c;传统行业面临着前所未有的挑战与机遇。产品管理作为连接市场需求与企业生产的核心环节&#xff0c;其重要性不言而喻。对于传统行业而言&#xff0c;做好产品管理不仅意味着保持竞争力&#xff0c;更是实现转型升级、拥抱未来的关键。…

【类模板】成员函数模板

一、成员函数模板的基本含义 不管是普通的类&#xff0c;还是类模板&#xff0c;都可以为其定义成员函数模板&#xff0c;以下的情况就是类模板和成员函数模板都有各自独立的一套参数&#xff1a; template<typename T1> class A { public:T1 m_ic;static constexpr int…

0.3 学习Stm32经历过的磨难

文章目录 用库函数传参 能否按位或STM32库函数XXX_GetFlagStatus和XXX_GetITStatus的区别 用库函数传参 能否按位或 答案是看清况&#xff0c;而不是一股脑的写&#xff01;&#xff08;血泪的经验啊&#xff09; 可行的情况&#xff1a; //如gpio初始化结构体中的gpiopin参…

c++list

list介绍 list是序列容器&#xff0c;允许对序列中任意位置的恒定时间插入和擦除操作&#xff0c;以及双向迭代。 list容器被实现为双向链表;双向链表可以将它们包含的每个元素存储在不同且不相关的存储位置。 list的使用 list中的接口比较多&#xff0c;此处类似&#xff0…

RedisStack十部曲之二:Redis的核心概念

文章目录 键空间修改和查询键键过期遍历键空间 客户端缓存在计算机科学中有两个难题客户端缓存的Redis实现跟踪模式的工作机制统一的键命名空间 两种连接方式缓存策略Opt-in 模式Opt-out 模式广播模式NOLOOP选项避免竟态条件当与服务器失去连接怎么办什么值得缓存 流水线请求/响…

使用QTcpSocket在两台ubuntu之间实现通讯

重点提取&#xff1a; 1.保证服务端和客户端端口号一致 2.保证服务端和客户端在同一网段(可以通过网线连接) 3保证客户端界面输入的ip是服务段的ip 实现步骤&#xff1a; 首先&#xff0c;构造服务端界面和客户端界面如下 服务端界面 客户端界面 其次具体代码 在.pro文件…

FRP内网穿透与神卓互联,优势对比

本文介绍分析了当前市面上两款常用的内网穿透工具 frp内网穿透介绍 一、概述 frp&#xff08;Fast Reverse Proxy&#xff09;是一款高性能的反向代理应用&#xff0c;主要用于实现内网穿透功能。通过frp&#xff0c;用户可以将内网中的服务器或服务暴露到公网上&#xff0c;…

【ACM独立出版|EI快检索-高录用|IEEE Fellow支持】2024年数字经济与计算机科学国际学术会议(DECS2024)

【ACM独立出版&#xff5c;EI快检索-高录用&#xff5c;IEEE Fellow支持】 2024年数字经济与计算机科学国际学术会议&#xff08;DECS2024&#xff09; *ACM独立出版&#xff0c;快检索&#xff0c;高录用 *见刊后1个月左右完成EI&Scopus检索 *国内211大学、世界QS名校…

#驱动开发

内核模块 字符设备驱动 中断、内核定时器 裸机开发和驱动开发的区别&#xff1f; 裸机开发 驱动开发&#xff08;基于内核&#xff09; 相同点 都能够控制硬件&#xff08;本质&#xff1a;操作寄存器&#xff09; 不同点 用C语言给对应的地址里面写值 按照一定的框架格式…

hackme靶机通关攻略

1、登录靶机&#xff0c;查询是否有注入点 2、判断闭合方式 输入OSINT and 11 # 输入OSINT and 12 # 得出闭合方式为单引号 2、查询数据库名 输入-1 union select database(),2,3 # 3、查询数据库中的表 输入-1 union select group_concat(table_name),2,3 from informa…

还在用谷歌翻译?这4款翻译工具也许更高效!

随着国内很多翻译工具的不断发展&#xff0c;谷歌翻译相对来说不是一款十分有优势的翻译工具。并且使用的时候还会受到网络的限制&#xff0c;如过大家有翻译方面的需求的话&#xff0c;不妨试试这几款翻译工具。不论是从翻译的语言种类&#xff0c;翻译质量还是翻译速度来看&a…