Java笔记 —— 反射(概念,2024年最新高级职位面试

  • 获取单个成员变量并修改变量的值

  • 获取成员方法

      • 批量获取成员方法
  • 获取单个成员方法,并调用

动态语言(比如JavaScript等)在运行时可以改变自身结构,增加新的函数或者对象。动态语言的在编写的时候不必指定变量的类型

静态语言(比如Java等)在运行时不能改变自身的结构,在编译期就要指定变量的类型。但是反射机制,让Java获得类似动态语言的特性,具有一定的动态性。

.java是源程序,是我们编写的代码,.class是编译后的程序,本质上是一个二进制文件。

类加载机制:

当程序需要用到某个类的时候,如果这个类还没有加载到内存中,那么Java虚拟机就会将这个类加载到内存中进行初始化。

具体初始化的步骤有:

  1. 加载:将.class文件读入内存(具体是在堆内存的方法区),并为之创建一个Class类型的对象

放一张廖雪峰老师网站的截图

廖雪峰老师网站的链接

在这里插入图片描述

这个String类的Class实例就会保存这个String类的全部信息

注意:

创建String类的语句是public final class String,而创建Class类的语句是public final class Class。这里的String是类名,Class也是类名。而String类的对象叫String类型对象,Class类的对象叫Class类型对象。不同的是,Class类是未知的,它可以是任意的类。就比如拿到一个class文件,我们不知道这个class文件里面是什么样子,有哪些变量和方法。那么就可以通过Class类型对象,来反方向获得这个class文件的类的全部信息。这个过程就叫做反射。

  1. 连接:具体又分为验证,准备,解析三个步骤。

(1)验证:对文件格式等内部结构是否正确

(2)准备:为类变量分配内存并设置初始值

(3)解析:将类的二进制数据中的符号引用替换为直接引用

  1. 初始化:到这一步开始执行类里面自己编写的代码,完成类的初始化

类加载器:负责将.class文件加载到内存中,并为之生成对应的Class对象。

反射的定义

加载完类之后产生的这个Class类型的对象(一个类只会有一个Class对象),会包含这个类的完整信息(包括类名,父类,接口,这个类的所有方法,变量等),我们通过这个Class对象看到这个对象对应的类的所有信息。这个对象就可以比喻成一面镜子,透过这面镜子看到类的信息。

这种通过Class对象获得其对应的类的所有信息的行为称为反射。(这里又总结一遍反射的定义)

类似于public final class String,String类一样,String类有构造方法和成员方法。

public final class Class是反射的核心类,也同样有着一套自己的构造方法和成员方法

反射的功能(包括但不限于):
  1. 在运行时判断任意一个对象所属的类

  2. 在运行时构造任意一个类的对象

  3. 在运行时判断任意一个类所具有的成员变量和方法

  4. 在运行时调用任意一个对象的成员变量和方法

像我之前说的例子,根据一个class文件对象去获得,使用这个class文件中成员方法,成员变量等,算是比较常用的功能之一。

获取Class类型对象

既然获取了一个class类的Class对象,就可以获得这个类的全部信息,那么我们该如何获得这个Class对象呢?

  1. 通过Object中的getClass()方法获取,返回该Object的运行时类

  2. 通过类名获取静态属性class

  3. Class类中的静态方法 public static 类<?> forName(String className)(通常使用这个方法

package test.reflexDemo;

public class Student {

private String name;

int age;

public String hobby;

public Student(){

}

private Student(String name){

this.name = name;

}

Student(String name,int age){

this.name = name;

this.age = age;

}

public Student(String name,int age,String hobby){

this.name = name;

this.age = age;

this.hobby = hobby;

}

public void test(){

System.out.println(“public修饰的无参的方法”);

}

public String show(String name,int age){

System.out.println(“public修饰的含参的方法”);

return name + “—” + age;

}

void function(String name){

System.out.println("默认修饰符的含参的方法,姓名是 "+name);

}

private void demo(String hobby){

System.out.println("private修饰的含参的方法,爱好是 "+hobby);

}

@Override

public String toString() {

return “Student{” +

“name='” + name + ‘’’ +

“, age=” + age +

“, hobby='” + hobby + ‘’’ +

‘}’;

}

}

这里的Student就相当于Class,这里的s就相当于Class类型对象

package test.reflexDemo;

public class Demo1 {

public static void main(String[] args) throws ClassNotFoundException {

//这里的Student就相当于Class,这里的s就相当于Class类型对象

Student s = new Student();

//方式一,getClass()方法,获得该对象的运行时类

Class<? extends Student> c1 = s.getClass();

Student s1 = new Student();

Class<? extends Student> c2 = s1.getClass();

System.out.println(c1==c2); // true

//这里结果为true的原因,同一个class可以new出来多个对象,但是只会对应一个Class类型对象

//所以c1和c2是一样的,但是s和s1是不一样的

System.out.println(s==s1); // false

System.out.println(“-----------------------------”);

//方式二,通过类名获得静态属性class

Class c3 = Student.class;

System.out.println(c3 == c1); //true

System.out.println(c3 == c2); //true

System.out.println(“-----------------------------”);

//方式三,利用Class类中的静态方法forName(String className)

//这个方法会返回给定字符串名称className的类或接口相关联的类对象,然后通过类对象调用类的方法和变量

//注意这里的className需要写类在项目中的具体位置

Class<?> c4 = Class.forName(“test.reflexDemo.Student”);

System.out.println(c4 == c3); //true

System.out.println(c4 == c2); //true

System.out.println(c4 == c1); //true

//Class后面的泛型可以不添加

//这样看起来更直接一些,c5是Class类型

Class c5 = Class.forName(“test.reflexDemo.Student”);

System.out.println(c5 == c4); //true

}

}

根据Class类型对象来创建类的实例

上面成功获得了Class类型对象,那么如何通过Class类型对象来创建对应的class类的对象呢?

思路是通过Class类型对象,来获得对应类的构造方法,然后再用构造方法来创建类的实例。

批量获取构造方法

package test.reflexDemo;

import java.lang.reflect.Constructor;

public class Demo2 {

public static void main(String[] args) throws ClassNotFoundException {

Class c = Class.forName(“test.reflexDemo.Student”);

//public Constructor<?>[] getConstructors()

// 返回文件中的所有public修饰的构造方法,不会获取private或者默认修饰的构造方法

Constructor[] constructors = c.getConstructors();

for (Constructor constructor : constructors) {

System.out.println(constructor);

}

System.out.println(“----------------------”);

//public Constructor<?>[] getDeclaredConstructors()

//返回所有的构造方法,包括private和默认修饰的构造方法

//另外这里获取的private修饰的构造方法也可以构建对象

Constructor[] dcs = c.getDeclaredConstructors();

for (Constructor d : dcs) {

System.out.println(d);

}

}

}

在这里插入图片描述

获取单个的构造方法

package test.reflexDemo;

import java.lang.reflect.Constructor;

public class Demo3 {

public static void main(String[] args) throws Exception {

Class c = Class.forName(“test.reflexDemo.Student”);

//public Constructor getConstructor(Class<?>… parameterTypes)

//括号里的参数是指具体数据类型对应的类,比如String要写成String.class

//什么都没写就是获取默认无参构造方法

Constructor cons1 = c.getConstructor();

System.out.println(“无参构造方法”+cons1);

Constructor cons2 = c.getConstructor(String.class,int.class,String.class);

System.out.println(“有参构造方法”+cons2);

//同样的,这样只能获得public类型,要想获得private类型需要改为

//getDeclaredConstructor

Constructor cons3 = c.getDeclaredConstructor(String.class);

System.out.println(“private修饰的有参构造方法”+cons3);

System.out.println(“--------------------------”);

//根据获得的构造方法创建实例

//public T newInstance(Object… initargs)

//括号里面写要传入构造方法中的参数

Object o1 = cons1.newInstance();

System.out.println(o1);

//这里可以向下转型

Student s = (Student)o1;

System.out.println(s);

//虽然获得了Private修饰的构造方法,但不能直接写入,会报权限不足的异常

//需要强行获取权限,暴力访问

cons3.setAccessible(true);

Object o2 = cons3.newInstance(“张”);

System.out.println(o2);

}

}

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
img
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip1024b (备注Java)
img

最后

如果觉得本文对你有帮助的话,不妨给我点个赞,关注一下吧!

一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!*

由于文件比较大,这里只是将部分目录大纲截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且后续会持续更新

如果你觉得这些内容对你有帮助,可以添加V获取:vip1024b (备注Java)
[外链图片转存中…(img-ZPELpQJL-1712794015178)]

最后

如果觉得本文对你有帮助的话,不妨给我点个赞,关注一下吧!

[外链图片转存中…(img-9asXXATb-1712794015178)]

[外链图片转存中…(img-Rl9leMUR-1712794015178)]

一个人可以走的很快,但一群人才能走的更远。不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎扫码加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
[外链图片转存中…(img-T6ssFnnf-1712794015178)]

  • 20
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
文件上传是Web开发中常见的功能之一,Java中也提供了多种方式来实现文件上传。其中,一种常用的方式是通过Apache的commons-fileupload组件来实现文件上传。 以下是实现文件上传的步骤: 1.在pom.xml文件中添加以下依赖: ```xml <dependency> <groupId>commons-fileupload</groupId> <artifactId>commons-fileupload</artifactId> <version>1.3.3</version> </dependency> ``` 2.在前端页面中添加文件上传表单: ```html <form method="post" enctype="multipart/form-data" action="upload"> <input type="file" name="file"> <input type="submit" value="Upload"> </form> ``` 3.在后台Java代码中处理上传文件: ```java // 创建一个DiskFileItemFactory对象,用于解析上传的文件 DiskFileItemFactory factory = new DiskFileItemFactory(); // 设置缓冲区大小,如果上传的文件大于缓冲区大小,则先将文件保存到临时文件中,再进行处理 factory.setSizeThreshold(1024 * 1024); // 创建一个ServletFileUpload对象,用于解析上传的文件 ServletFileUpload upload = new ServletFileUpload(factory); // 设置上传文件的大小限制,这里设置为10MB upload.setFileSizeMax(10 * 1024 * 1024); // 解析上传的文件,得到一个FileItem的List集合 List<FileItem> items = upload.parseRequest(request); // 遍历FileItem的List集合,处理上传的文件 for (FileItem item : items) { // 判断当前FileItem是否为上传的文件 if (!item.isFormField()) { // 获取上传文件的文件名 String fileName = item.getName(); // 创建一个File对象,用于保存上传的文件 File file = new File("D:/uploads/" + fileName); // 将上传的文件保存到指定的目录中 item.write(file); } } ``` 以上代码中,首先创建了一个DiskFileItemFactory对象,用于解析上传的文件。然后设置了缓冲区大小和上传文件的大小限制。接着创建一个ServletFileUpload对象,用于解析上传的文件。最后遍历FileItem的List集合,判断当前FileItem是否为上传的文件,如果是,则获取文件名,创建一个File对象,将上传的文件保存到指定的目录中。 4.文件上传完成后,可以给用户一个提示信息,例如: ```java response.getWriter().write("File uploaded successfully!"); ``` 以上就是使用Apache的commons-fileupload组件实现文件上传的步骤。需要注意的是,文件上传可能会带来安全隐患,因此在处理上传的文件时,需要进行严格的校验和过滤。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值