什么是反射
**反射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等方法,可以判断是否包含某些修饰符。