反射

一.什么是反射?用反射做什么?

反射机制指的是程序在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法。

反射是:指程序可以访问、检测和修改它本身状态或行为的一种能力

反射是一种能力,所以给的定义就是说明了它能干嘛。

我们平时用反射主要做:
获取类型的相关信息
动态调用方法
动态构造对象
从程序集中获得类型

二:反射的优缺点

优点:
反射提高了程序的灵活性和扩展性,降低耦合性,提高自适应能力。它允许程序创建和控制任何类的对象,无需提前硬编码目标类;

缺点:
1、性能问题:使用反射基本上是一种解释操作,用于字段和方法接入时要远慢于直接代码。因此反射机制主要应用在对灵活性和扩展性要求很高的系统框架上,普通程序不建议使用。
2、使用反射会模糊程序内内部逻辑:程序员希望在源代码中看到程序的逻辑,反射等绕过了源代码的技术,因而会带来维护问题。反射代码比相应的直接代码更复杂。 至于执行效率的话,还可以,因为它是一种强类型语言,执行效率不错。不过,建议将反射过后,保存进 cache中。
根据Student类进行反射的示例

package com.ningjie;

public class Student {
	private String sid;

	private String sname;

	public Integer age;
	
	static{
		System.out.println("加载进jvm中!");
	}

	public Student() {
		super();
		System.out.println("调用带一个参数的构造方法创建了一个学生对象");
	}

	public Student(String sid) {
		super();
		this.sid = sid;
		System.out.println("调用带一个参数的构造方法创建了一个学生对象");	}

	public Student(String sid, String sname) {
		super();
		this.sid = sid;
		this.sname = sname;
		System.out.println("调用带二个参数的构造方法创建了一个学生对象");
	}

	@SuppressWarnings("unused")
	private Student(Integer age) {
		System.out.println("调用Student类私有的构造方法创建一个学生对象");
		this.age = age;
	}

	public String getSid() {
		return sid;
	}

	public void setSid(String sid) {
		this.sid = sid;
	}

	public String getSname() {
		return sname;
	}

	public void setSname(String sname) {
		this.sname = sname;
	}

	public void hello() {
		System.out.println("你好!我是" + this.sname);
	}

	public void hello(String name) {
		System.out.println(name + "你好!我是" + this.sname);
	}

	@SuppressWarnings("unused")
	private Integer add(Integer a, Integer b) {
		return new Integer(a.intValue() + b.intValue());
	}
}

三:反射中如何获取类对象

1.Class.forName(“类的全路径名”); jdbc,自定义mvc框架
2.类名.class 做通用的查询
3.类(Class类类的类对象)实例.getClass() 做同用的增删改

package com.ningjie;
/**
 * 如何获取类对象
 * 1.Class.forName("类的全路径名");   jdbc,自定义mvc框架
 * 2.类名.class   做通用的查询
 * 3.类(Class类类的类对象)实例.getClass()    做同用的增删改
 * @author NJ321
 *
 */
public class Demo1 {
public static void main(String[] args) throws ClassNotFoundException {
	//方法一:Class.forName("类的全路径名")
	//Class<?> clz = Class.forName("com.ningjie.Student");
	//System.out.println(clz);
	//方法二:类名.class
	//Class<?> clz=Student.class;
	//System.out.println(clz);
	//方法三:类实例.getClass()
	//Student stu=new Student();
	//Class<?> clz=stu.getClass();
	//System.out.println(clz);
	/**
	 * 为什么jdbc链接要使用Class.forName("com.jdbc.mysql.Driver");
	 * 无论是mysql的驱动Driver的驱动......;他必须需要实现jdbc的一个驱动接口
	 * com.jdbc.mysql.Driver extends java.sql.Driver
	 * java.sql.Driver d=Class.forName("com.jdbc.mysql.Driver");
	 */
}
}

方法一:Class.forName(“类的全路径名”)

Class<?> clz = Class.forName("com.ningjie.Student");
System.out.println(clz);

运行结果:
在这里插入图片描述
方法二:类名.class

Class<?> clz=Student.class;
System.out.println(clz);

运行结果:
在这里插入图片描述
方法三:类实例.getClass()

Student stu=new Student();
Class<?> clz=stu.getClass();
System.out.println(clz);

运行结果:
在这里插入图片描述

四:反射实例化

反射实例化的好处
1.能够实例化未知的类
2.能够通过私有构造器实例化

package com.ningjie;

import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;

/**
 * 反射实例化
 * 1.能够实例化未知的类
 * 2.能够通过私有构造器实例化
 * @author NJ321
 *
 */
public class Demo2 {
public static void main(String[] args) throws InstantiationException, IllegalAccessException, NoSuchMethodException, SecurityException, IllegalArgumentException, InvocationTargetException {
	Class clz=Student.class;
	//反射调用无参构造方法创建了一个学生对象
	//Student stu=(Student)clz.newInstance();
	
	
	//调用带一个参数的构造方法创建了一个学生对象
	//要拿到构造类
	//Constructor con = clz.getConstructor(String.class);
	//通过构造器实例化对象
	//Student stu=(Student)con.newInstance("s001");
	
	
	//调用带二个参数的构造方法创建了一个学生对象
	//要拿到构造类
	//Constructor con = clz.getConstructor(String.class,String.class);
	//通过构造器实例化对象
	//Student stu=(Student)con.newInstance("s001","zs");
	
	
	//java.lang.NoSuchMethodException
	//getConstructor只能寻找到public修饰的构造器,getDeclaredConstructor可以寻找到任何修饰符修饰的构造器
	//Constructor con = clz.getConstructor(Integer.class);
	Constructor con = clz.getDeclaredConstructor(Integer.class);
	con.setAccessible(true);
		//通过构造器实例化对象
	  Student stu=(Student)con.newInstance(12);
	
}
}

1.反射调用无参构造方法创建了一个学生对象

Class clz=Student.class;
Student stu=(Student)clz.newInstance();

2.通过构造器实例化对象
1.调用带一个参数的构造方法创建了一个学生对象

Class clz=Student.class;
//调用带一个参数的构造方法创建了一个学生对象
//要拿到构造类
Constructor con = clz.getConstructor(String.class);
//通过构造器实例化对象
Student stu=(Student)con.newInstance("s001");

运行结果:
在这里插入图片描述
2.调用带两个参数的构造方法创建了一个学生对象

//调用带二个参数的构造方法创建了一个学生对象
//要拿到构造类
Constructor con =clz.getConstructor(String.class,String.class);
//通过构造器实例化对象
Student stu=(Student)con.newInstance("s001","zs");

运行结果:
在这里插入图片描述
3.调用Student类私有的构造方法创建一个学生对象
注:getConstructor只能寻找到public修饰的构造器,getDeclaredConstructor可以寻找到任何修饰符修饰的构造器。(使用getDeclaredConstructor时,必须调用setAccessible方法,并且设置为true方可进行访问

Class clz=Student.class;
Constructor con = clz.getDeclaredConstructor(Integer.class);
con.setAccessible(true);
 Student stu=(Student)con.newInstance(12);

运行结果:
在这里插入图片描述

五:反射调用方法

1:

Student stu=new Student();
Class clz=stu.getClass();
Method m = clz.getDeclaredMethod("hello");
m.invoke(stu);

运行结果:
在这里插入图片描述
2:

Student stu=new Student();
Class clz=stu.getClass();
Method m = clz.getDeclaredMethod("hello",String.class);
m.invoke(stu,"zs");

运行结果:
在这里插入图片描述
3:

Student stu=new Student();
Class clz=stu.getClass();
Method m=clz.getDeclaredMethod("add",Integer.class,Integer.class);
m.setAccessible(true);
//invoke:如果反射动态调用的方法是被void所修饰,那么返回的就是null
//如果反射动态调用的方法不是被void所修饰,那么返回的就是被调用的方法的返回值
Object invoke = m.invoke(stu,20,5);
System.out.println(invoke);

运行结果:
在这里插入图片描述

六:反射属性赋值取值

反射能够将jsp传递来的参数直接封装到实体类中

package com.ningjie;

import java.lang.reflect.Field;

/**
 * 反射属性赋值取值
 * 反射能够将jsp传递来的参数直接封装到实体类中
 * @author NJ321
 *
 */
public class Demo4 {
public static void main(String[] args) throws NoSuchFieldException, SecurityException, IllegalArgumentException, IllegalAccessException {
	Student stu=new Student("s002","ls");
	stu.age=22;
	Class clz=stu.getClass();
	//Field field = clz.getDeclaredField("age");
	//Field field = clz.getDeclaredField("sid");
	//field.setAccessible(true);
	//field.set(stu,"s002");
	//System.out.println(stu);
	//System.out.println(field.get(stu));
	Field[] fields = clz.getDeclaredFields();
	for (Field field : fields) {
		field.setAccessible(true);
		System.out.println(field.getName()+":"+field.get(stu));
	}
}
}

1.属性值得读写

	Student stu=new Student();
	Class clz=stu.getClass();
	//Field field = clz.getDeclaredField("age");
	Field field = clz.getDeclaredField("sid");
	field.setAccessible(true);
	field.set(stu,"s002");
	System.out.println(stu);
	System.out.println(field.get(stu));

运行结果:

在这里插入图片描述
2.如果实体类中属性过多,可用以下方法调用

Student stu=new Student("s002","ls");
stu.age=22;
Class clz=stu.getClass();
Field[] fields = clz.getDeclaredFields();
for (Field field : fields) {
	field.setAccessible(true);
	System.out.println(field.getName()+":"+field.get(stu));
	}

运行结果:
在这里插入图片描述

七:访问修饰符

在Java中提供了四种访问权限,使用不同的访问权限时,被修饰的内容会有不同的访问权限,以下表来说明不同权限的访问能力:
在这里插入图片描述
l 要想仅能在本类中访问使用private修饰;

l 要想本包中的类都可以访问不加修饰符即可;

l 要想本包中的类与其他包中的子类可以访问使用protected修饰

l 要想所有包中的所有类都可以访问使用public修饰。

l 注意:如果类用public修饰,则类名必须与文件名相同。一个文件中只能有一个public修饰的类。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值