反射

什么是反射

 **反射java语言中的一种机制,通过这种机制可以动态的实例化对象、读写属性、调用方法**

借用网上的一张反射关系图:
在这里插入图片描述
反射的优点和缺点:
优点:

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

1 性能问题:使用反射基本上是一种解释操作,用于字段和方法接入时要远慢于直接代码。因此反射机制主要应用在对灵活性和拓 展性要求很高的系统框架上,普通程序不建议使用。

2 使用反射会模糊程序内部逻辑;程序员希望在源代码中看到程序的逻辑,反射却绕过了源代码的技术,因而会带来维护的问题,反射代码比相应的直接代码更复杂

反射三大作用
1 实例化对象

  c.newInstance()
  Constructor.getConstructor/Constructor.getDeclaredConstructor  
  注:一定要提供无参构造器

2 动态调用方法

  Method m;
  m.invoke

3 读写属性

  Field set/get

实例化对象

案例资料:

package com.zhoutubing.reflect;

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());
	}
}

一切反射相关的代码都从获得类(java.lang.Class)对象开始
1 Class.forName(完整类名)

package com.zhoutubing.reflect;

/**
 *  如何获取类对象
 *    1 Class.forName("类的全名路径");   jdbc 自定义mvc框架
 * @author Administrator
 *
 */
public class Demo1 {
	public static void main(String[] args) throws ClassNotFoundException {
				Class<?> clz = Class.forName("com.zhoutubing.reflect.Student");
	}
}

结果:
在这里插入图片描述
2 类名.class

package com.zhoutubing.reflect;

/**
 *  如何获取类对象
 *    2 类名.class   做通用的查询
 * @author Administrator
 *
 */
public class Demo1 {

	public static void main(String[] args) throws ClassNotFoundException {
		        Class<?> clz =Student.class;
		        System.out.println(clz);
	}
}

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

3 对象.getClass()

package com.zhoutubing.reflect;

/**
 *  如何获取类对象
 *    3 类(Class类类的类对象) 实例.getClass()   做通用的增删改
 * @author Administrator
 *
 */
public class Demo1 {

	public static void main(String[] args) throws ClassNotFoundException {
		        Student stu= new Student();
		        Class<?> clz = stu.getClass();
		        System.out.println(clz);
	}
}

结果:
在这里插入图片描述
上课问题:
为什么jdbc要使用Class.forName(“com.jdbc.mysql.Driver”);

 无论是mysql的驱动Driver还是Oracle的驱动...;它必然需要实现jdbc的一个驱动接口;
 com.jdbc.mysql.Driver extends java.sql.Driver
 java.sql.Driver d = Class.forName("com.jdbc.mysql.Driver");

反射实例化

代码:

package com.zhoutubing.reflect;

import java.lang.reflect.Constructor;

/**
 * 反射实例化
 *  1 能够实例化未知的类
 *  2 能够通过私有构造器创建实例
 * @author Administrator
 *
 */
public class Demo2 {

	public static void main(String[] args) throws Exception {
		Class clz = Student.class;
//		调用无参构造方法创建了一个学生对象
//		Object 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);
	}
}

反射调用方法

package com.zhoutubing.reflect;

import java.lang.reflect.Method;

/**
 * 反射调用方法
 *   AddOrderServlet
 *   DelOrderServlet
 *   
 *   -->OrderServlet
 *   jsp-->methodName
 *   dopost
 *   String methodName = req.getParameter("methodName");
 *   if("add".equals(methodName)){
 *   
 *   }else if("del".equals(methodName)){
 *      
 *   }
 *   --->
 *   上面的代码全部都不需要了
 *   del(req,resp);
 *   add(req,resp);
 * @author Administrator
 *
 */
public class Demo3 {

	public static void main(String[] args) throws Exception{
//		正常调用方法
		Student stu = new Student();
		stu.hello();
		Class clz = stu.getClass();
		Method m = clz.getDeclaredMethod("hello");
		m.invoke(stu);
	}
}

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

package com.zhoutubing.reflect;

import java.lang.reflect.Method;

/**
 * 反射调用方法
 *   AddOrderServlet
 *   DelOrderServlet
 *   
 *   -->OrderServlet
 *   jsp-->methodName
 *   dopost
 *   String methodName = req.getParameter("methodName");
 *   if("add".equals(methodName)){
 *   
 *   }else if("del".equals(methodName)){
 *      
 *   }
 *   --->
 *   上面的代码全部都不需要了
 *   del(req,resp);
 *   add(req,resp);
 * @author Administrator
 *
 */
public class Demo3 {

	public static void main(String[] args) throws Exception{
	    Student stu = new Student();
		Class clz = stu.getClass();
		Method m = clz.getDeclaredMethod("hello",String.class);
		m.invoke(stu,"zs");
	}
}

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

package com.zhoutubing.reflect;

import java.lang.reflect.Method;

/**
 * 反射调用方法
 *   AddOrderServlet
 *   DelOrderServlet
 *   
 *   -->OrderServlet
 *   jsp-->methodName
 *   dopost
 *   String methodName = req.getParameter("methodName");
 *   if("add".equals(methodName)){
 *   
 *   }else if("del".equals(methodName)){
 *      
 *   }
 *   --->
 *   上面的代码全部都不需要了
 *   del(req,resp);
 *   add(req,resp);
 * @author Administrator
 *
 */
public class Demo3 {

	public static void main(String[] args) throws Exception{
	    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);
	}
}

结果:

在这里插入图片描述

package com.zhoutubing.reflect;

import java.lang.reflect.Method;

/**
 * 反射调用方法
 *   AddOrderServlet
 *   DelOrderServlet
 *   
 *   -->OrderServlet
 *   jsp-->methodName
 *   dopost
 *   String methodName = req.getParameter("methodName");
 *   if("add".equals(methodName)){
 *   
 *   }else if("del".equals(methodName)){
 *      
 *   }
 *   --->
 *   上面的代码全部都不需要了
 *   del(req,resp);
 *   add(req,resp);
 * @author Administrator
 *
 */
public class Demo3 {

	public static void main(String[] args) throws Exception{
		Student stu = new Student();
		Class clz = stu.getClass();
		
		Method m = clz.getDeclaredMethod("hello",String.class);
		System.out.println(m.invoke(stu, "zs"));
	}
}

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

反射属性赋值取值

代码:

package com.zhoutubing.reflect;

import java.lang.reflect.Field;

/**
 * 反射属性赋值取值
 *  jsp
 *    -->uname,sex,age,password,phone,address...
 *    Servlet
 *    String uanme = req.getParameter("uname");
 *    String sex = req.getParameter("sex");
 *    ....
 *    User u = new User();
 *    u.setName(uname);
 *    u.setSex(sex);
 *    ...
 *    
 *    -->
 *    反射能够将jsp传递过来的参数直接封装到实体类中
 * @author Administrator
 *
 */
public class Demo4 {

	public static void main(String[] args) throws Exception{
		Student stu = new Student("s002","zs");
//		stu.setSid("s001");
		stu.age=23;
//		System.out.println(stu);
		
		Class clz = stu.getClass();
//		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));
		}
	}
}

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

访问修饰符

如何判断类或变量、方法的修饰符,可以使用Java反射机制中,Field的 getModifiers() 方法返回int类型值表示该字段的修饰符,
即这个方法就是返回一个int型的返回值,代表类、成员变量、方法的修饰符。
其中,该修饰符是java.lang.reflect.Modifier的静态属性。

对应表如下:
PUBLIC: 1
PRIVATE: 2
PROTECTED: 4
STATIC: 8
FINAL: 16
SYNCHRONIZED: 32
VOLATILE: 64
TRANSIENT: 128
NATIVE: 256
INTERFACE: 512
ABSTRACT: 1024
STRICT: 2048

Modifier提供了很多静态方法。如:
public static String toString(int mod) 可以输出该整数对应的所有的修饰符。
public static boolean isPublic(int mod)可以判断该整数对应的是不是包含public修饰符。
通过Modifier的isPublic、isPrivate、isStatic等方法,可以判断是否包含某些修饰符。

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值