根据实例详解Java中的反射机制

简介: 概念:JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。作用:Java反射机制主要提供了以下功能: 在运行时判断任意一个对象所属的类;在运行时构造任意一个类的对象;在运行时判断任意一个类所具有的成员变量和方法;在运行时调用任意一个对象
概念:
JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。
作用:
Java反射机制主要提供了以下功能: 在运行时判断任意一个对象所属的类;在运行时构造任意一个类的对象;在运行时判断任意一个类所具有的成员变量和方法;在运行时调用任意一个对象的方法;生成动态代理。

可以说Java的反射机制是Java中所有框架的基础,要对SSH等框架有一个深刻的理解,必须熟悉掌握Java的反射机制。

下面我们通过具体例子来了解一下Java反射机制的原理及使用

1, 通过对象获取对象所属的包名和类名

在demo2包中定义一个类:

package demo2;
public class ReflectDemo2 {
	
}
获取包名加类名:

public class ReflectDemo1 {
	public static void main(String[] args) {
		ReflectDemo2 rd=new ReflectDemo2();
		System.out.println(rd.getClass().getName());
	}
}
运行结果:demo2.ReflectDemo2

2,获得Class对象的三种方法:

public class ReflectDemo1 {
	public static void main(String[] args) {
		Class<?> c1 = null;
		Class<?> c2 = null;
		Class<?> c3 = null;
		try {
			c1 = Class.forName("demo2.ReflectDemo2");
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		}
		System.out.println("c1:"+c1.getName());
		
		c2 = new ReflectDemo2().getClass();
		System.out.println("c2:"+c2.getName());
		
		c3 = ReflectDemo2.class;
		System.out.println("c3:"+c3.getName());
	}
}
运行结果:

c1:demo2.ReflectDemo2
c2:demo2.ReflectDemo2
c3:demo2.ReflectDemo2

我们可以看到,运行的结果都是一样的,c1通过类Class的静态方法forName();c2通过实例变量的getClass()方法;c3直接通过对象类的.class文件获得,使用比较普遍的是通过Class的静态方法forName()获取对象。

3, 通过反射机制实例化类的对象
首先创建对象类:

public class User {
	private String username;
	private String password;
	public String getUsername() {
		return username;
	}
	public void setUsername(String username) {
		this.username = username;
	}
	public String getPassword() {
		return password;
	}
	public void setPassword(String password) {
		this.password = password;
	}
	public User(String username, String password) {
		super();
		this.username = username;
		this.password = password;
	}
	public User() {
		super();
		// TODO Auto-generated constructor stub
	}
	@Override
	public String toString() {
		return "User [username=" + username + ", password=" + password + "]";
	}
	
}

一,使用set方法为实例化对象赋值:

public class ReflectDemo1 {
	public static void main(String[] args) throws ClassNotFoundException {
		Class<?> c1 = Class.forName("demo2.User");
		try {
			User user = (User)c1.newInstance();
			user.setUsername("张三");
			user.setPassword("zhangsan");
			System.out.println(user);
		} catch (InstantiationException e) {
			e.printStackTrace();
		} catch (IllegalAccessException e) {
			e.printStackTrace();
		}
		
	}
}
运行结果:
User [username=张三, password=zhangsan]

二,使用构造函数方法为实例化对象赋值:

public class ReflectDemo1 {
	public static void main(String[] args) throws ClassNotFoundException {
		Class<?> c1 = Class.forName("demo2.User");
		//取得全部的构造函数
        Constructor<?> cons[] = c1.getConstructors();
        try {
			User user = (User)cons[0].newInstance("李四","123");
			System.out.println(user);
		} catch (IllegalArgumentException e) {
			e.printStackTrace();
		} catch (InstantiationException e) {
			e.printStackTrace();
		} catch (IllegalAccessException e) {
			e.printStackTrace();
		} catch (InvocationTargetException e) {
			e.printStackTrace();
		}
	}
}

运行结果:
User [username=李四, password=123]

4, 获取类的全部属性信息:

public class ReflectDemo2{
	public static void main(String[] args) {
		try {
			Class<?> c1 = Class.forName("demo2.User");
			Field[] field = c1.getDeclaredFields();
			for (int i = 0; i < field.length; i++) {
				int p = field[i].getModifiers();//获取属性的作用域
				String p1 = Modifier.toString(p);
				Class<?> type = field[i].getType();//获取属性的类型
				System.out.println("属性名:"+field[i].getName()+"作用域是:"+p1+"类型是:"+type.getName());
			}
		} catch (ClassNotFoundException e) {
			e.printStackTrace();
		}
		
	}	
}

运行结果:
属性名:serialVersionUID作用域是:private static final类型是:long
属性名:username作用域是:private类型是:java.lang.String
属性名:password作用域是:private类型是:java.lang.String

5.获取某个类的全部方法:

public class ReflectDemo2{
	public static void main(String[] args) {
			Class<?> c1;
			try {
				c1 = Class.forName("demo2.User");
				Method method[] = c1.getMethods();
		        for (int i = 0; i < method.length; ++i) {
		            System.out.print(method[i].getName() + " ");
		        }
			} catch (ClassNotFoundException e) {
				e.printStackTrace();
			}
	}	
}

运行结果:
toString getUsername setUsername getPassword setPassword wait wait wait hashCode getClass equals notify notifyAll

6, 通过反射机制调用某个类的方法

首先定义要调用的类:

public class User implements Serializable{
	private static final long serialVersionUID = 7331750553985640492L;
	private String username;
	private String password;
	public String getUsername() {
		return username;
	}
	public void setUsername(String username) {
		this.username = username;
	}
	public String getPassword() {
		return password;
	}
	public void setPassword(String password) {
		this.password = password;
	}
	public User(String username, String password) {
		super();
		this.username = username;
		this.password = password;
	}
	public User(String username) {
		this.username = username;
	}
	public User() {
		super();
		// TODO Auto-generated constructor stub
	}
	@Override
	public String toString() {
		return "User [username=" + username + ", password=" + password + "]";
	}
	public void test1(){
		System.out.println("无参方法");
	}
	public void test2(String name){
		System.out.println("带参数的方法,名字为:"+name);
	}
}

通过反射机制调用User类中的方法:

public class ReflectDemo2{
	public static void main(String[] args) throws Exception {
			Class<?> c1 = Class.forName("demo2.User");
			//调用User类中的无参方法
			Method method = c1.getMethod("test1");
			method.invoke(c1.newInstance());
			//调用User类中的带参数的方法
			method = c1.getMethod("test2",String.class);
			method.invoke(c1.newInstance(),"阿木侠");
	}	
}

运行结果:
无参方法
带参数的方法,名字为:阿木侠

7,操作某个类的属性,对某个类的变量动态赋值:

public class ReflectDemo2{
	public static void main(String[] args) throws Exception {
			Class<?> c1 = Class.forName("demo2.User");
			Object obj = c1.newInstance();
			Field field = c1.getDeclaredField("username");
	        field.setAccessible(true);
	        field.set(obj, "这里对User类中的username进行赋值啦");
	        System.out.println(field.get(obj));
	}	
}
运行结果:
这里对User类中的username进行赋值啦

8, 通过Java的反射机制取得数组的信息,修改数组的信息

public class Test {
	 public static void main(String[] args) throws Exception {
	        int[] arr = { 1, 2, 3, 4, 5 };
	        Class<?> demo = arr.getClass().getComponentType();
	        System.out.println("数组类型: " + demo.getName());
	        System.out.println("数组长度  " + Array.getLength(arr));
	        System.out.println("数组的第一个元素: " + Array.get(arr, 0));
	        Array.set(arr, 0, 6);
	        System.out.println("修改之后数组第一个元素为: " + Array.get(arr, 0));
	    }
}
运行结果:
数组类型: int
数组长度  5
数组的第一个元素: 1
修改之后数组第一个元素为: 6

9, 工厂模式中Java反射机制的应用
在不使用反射机制的普通工厂类中,我们每次添加子类时都要对工厂类进行修改,很不方便,而利用反射机制实现工厂模式,可以在不修改工厂类的情况下添加任意多个子类,例如:
定义People接口:

public interface People {
	public abstract void read();
}
用学生和工人实现这个接口:

public class Worker implements People{

	public void read() {
		System.out.println("工人爱读书");
	}
}
public class Student implements People {

	public void read() {
		System.out.println("学生爱读书");
	}

}
定义工厂类:

public class FactoryDemo {
	public static People getInstance(String name) throws Exception{
		People p;
		p = (People)Class.forName(name).newInstance();
		return p;
		
	}
}

测试这个工厂类是否能实现:

public class ReflectDemo2{
	public static void main(String[] args) throws Exception {
			People student = FactoryDemo.getInstance("demo2.Student");
			if(student!=null){
				student.read();
			}
			People worker = FactoryDemo.getInstance("demo2.Worker");
			if(worker!=null){
				worker.read();
			}
	}	
}
运行结果:
学生爱读书
工人爱读书







目录
相关文章
|
21天前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
56 2
|
26天前
|
Java 编译器
探索Java中的异常处理机制
【10月更文挑战第35天】在Java的世界中,异常是程序运行过程中不可避免的一部分。本文将通过通俗易懂的语言和生动的比喻,带你了解Java中的异常处理机制,包括异常的类型、如何捕获和处理异常,以及如何在代码中有效地利用异常处理来提升程序的健壮性。让我们一起走进Java的异常世界,学习如何优雅地面对和解决问题吧!
|
1月前
|
XML 安全 Java
Java反射机制:解锁代码的无限可能
Java 反射(Reflection)是Java 的特征之一,它允许程序在运行时动态地访问和操作类的信息,包括类的属性、方法和构造函数。 反射机制能够使程序具备更大的灵活性和扩展性
41 5
Java反射机制:解锁代码的无限可能
|
25天前
|
Java 数据库连接 开发者
Java中的异常处理机制及其最佳实践####
在本文中,我们将探讨Java编程语言中的异常处理机制。通过深入分析try-catch语句、throws关键字以及自定义异常的创建与使用,我们旨在揭示如何有效地管理和响应程序运行中的错误和异常情况。此外,本文还将讨论一些最佳实践,以帮助开发者编写更加健壮和易于维护的代码。 ####
|
1月前
|
安全 IDE Java
Java反射Reflect机制详解
Java反射(Reflection)机制是Java语言的重要特性之一,允许程序在运行时动态地获取类的信息,并对类进行操作,如创建实例、调用方法、访问字段等。反射机制极大地提高了Java程序的灵活性和动态性,但也带来了性能和安全方面的挑战。本文将详细介绍Java反射机制的基本概念、常用操作、应用场景以及其优缺点。 ## 基本概念 ### 什么是反射 反射是一种在程序运行时动态获取类的信息,并对类进行操作的机制。通过反射,程序可以在运行时获得类的字段、方法、构造函数等信息,并可以动态调用方法、创建实例和访问字段。 ### 反射的核心类 Java反射机制主要由以下几个类和接口组成,这些类
60 2
|
1月前
|
存储 缓存 安全
🌟Java零基础:深入解析Java序列化机制
【10月更文挑战第20天】本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
29 3
|
1月前
|
安全 Java UED
深入理解Java中的异常处理机制
【10月更文挑战第25天】在编程世界中,错误和意外是不可避免的。Java作为一种广泛使用的编程语言,其异常处理机制是确保程序健壮性和可靠性的关键。本文通过浅显易懂的语言和实际示例,引导读者了解Java异常处理的基本概念、分类以及如何有效地使用try-catch-finally语句来处理异常情况。我们将从一个简单的例子开始,逐步深入到异常处理的最佳实践,旨在帮助初学者和有经验的开发者更好地掌握这一重要技能。
28 2
|
1月前
|
Java 数据库连接 开发者
Java中的异常处理机制####
本文深入探讨了Java语言中异常处理的核心概念,通过实例解析了try-catch语句的工作原理,并讨论了finally块和throws关键字的使用场景。我们将了解如何在Java程序中有效地管理错误,提高代码的健壮性和可维护性。 ####
|
1月前
|
Java
Java中的反射机制与应用实例
【10月更文挑战第22天】Java作为一门面向对象的编程语言,提供了丰富的特性来支持对象的创建、操作和交互。其中,反射机制是Java的一项核心特性,它允许程序在运行时动态地获取类的信息、创建对象、调用方法、访问属性等。本文将从三个部分探讨Java中的反射机制及其应用实例:一是反射机制的基本概念和原理;二是反射机制在Java中的应用场景;三是通过实例深入理解反射机制的使用方法和技巧。
24 4
|
2月前
|
安全 Java 程序员
深入浅出Java中的异常处理机制
【10月更文挑战第20天】本文将带你一探Java的异常处理世界,通过浅显易懂的语言和生动的比喻,让你在轻松阅读中掌握Java异常处理的核心概念。我们将一起学习如何优雅地处理代码中不可预见的错误,确保程序的健壮性和稳定性。准备好了吗?让我们一起踏上这段旅程吧!
27 6