1.java动态代理、反射(IDEA导入JUnit4)
1.1.反射
通过反射的方式可以获取class对象中的属性、方法、构造函数等
1.2.反射代码
import java.io.Serializable; public class Person implements Serializable,TestInterface{ private Long id; public String name; public Person() { this.id = 100L; this.name = "afsdfasd"; } public Person(Long id, String name) { //super(); this.id = id; this.name = name; } public Person(Long id) { super(); this.id = id; } @SuppressWarnings("unused") private Person(String name) { super(); this.name = name+"======="; } public Long getId() { return id; } public void setId(Long id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String toString() { return "Person [id=" + id + ", name=" + name + "]"; } private String getSomeThing() { return "sdsadasdsasd"; } private void testPrivate(){ System.out.println("this is a private method"); } } package cn.itcast_04_reflect; import java.io.Serializable; public class Person implements Serializable,TestInterface{ private Long id; public String name; public Person() { this.id = 100L; this.name = "afsdfasd"; } public Person(Long id, String name) { //super(); this.id = id; this.name = name; } public Person(Long id) { super(); this.id = id; } @SuppressWarnings("unused") private Person(String name) { super(); this.name = name+"======="; } public Long getId() { return id; } public void setId(Long id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String toString() { return "Person [id=" + id + ", name=" + name + "]"; } private String getSomeThing() { return "sdsadasdsasd"; } private void testPrivate(){ System.out.println("this is a private method"); } }
package cn.itcast_04_reflect; import java.lang.reflect.Constructor; import java.lang.reflect.Field; import java.lang.reflect.Method; import java.util.ArrayList; import java.util.List; import org.junit.Before; import org.junit.Test; public class MyReflect { public String className = null; @SuppressWarnings("rawtypes") public Class personClass = null; /** * 反射Person类 * @throws Exception */ @Before public void init() throws Exception { className = "cn.itcast_04_reflect.Person"; personClass = Class.forName(className); } /** *获取某个class文件对象 */ @Test public void getClassName() throws Exception { System.out.println(personClass); } /** *获取某个class文件对象的另一种方式 */ @Test public void getClassName2() throws Exception { System.out.println(Person.class); } /** *创建一个class文件表示的实例对象,底层会调用空参数的构造方法 */ @Test public void getNewInstance() throws Exception { System.out.println(personClass.newInstance()); } /** *获取非私有的构造函数 */ @SuppressWarnings({ "rawtypes", "unchecked" }) @Test public void getPublicConstructor() throws Exception { Constructor constructor = personClass.getConstructor(Long.class,String.class); Person person = (Person)constructor.newInstance(100L,"zhangsan"); System.out.println(person.getId()); System.out.println(person.getName()); } /** *获得私有的构造函数 */ @SuppressWarnings({ "rawtypes", "unchecked" }) @Test public void getPrivateConstructor() throws Exception { Constructor con = personClass.getDeclaredConstructor(String.class); con.setAccessible(true);//强制取消Java的权限检测 Person person2 = (Person)con.newInstance("zhangsan"); System.out.println("**"+person2.getName()); } /** *访问非私有的成员变量 */ @SuppressWarnings({ "rawtypes", "unchecked" }) @Test public void getNotPrivateField() throws Exception { Constructor constructor = personClass.getConstructor(Long.class,String.class); Object obj = constructor.newInstance(100L,"zhangsan"); Field field = personClass.getField("name"); field.set(obj, "lisi"); System.out.println(field.get(obj)); } /** *访问私有的成员变量 */ @SuppressWarnings({ "rawtypes", "unchecked" }) @Test public void getPrivateField() throws Exception { Constructor constructor = personClass.getConstructor(Long.class); Object obj = constructor.newInstance(100L); Field field2 = personClass.getDeclaredField("id"); field2.setAccessible(true);//强制取消Java的权限检测 field2.set(obj,10000L); System.out.println(field2.get(obj)); } /** *获取非私有的成员函数 */ @SuppressWarnings({ "unchecked" }) @Test public void getNotPrivateMethod() throws Exception { System.out.println(personClass.getMethod("toString")); Object obj = personClass.newInstance();//获取空参的构造函数 Method toStringMethod = personClass.getMethod("toString"); Object object = toStringMethod.invoke(obj); System.out.println(object); } /** *获取私有的成员函数 */ @SuppressWarnings("unchecked") @Test public void getPrivateMethod() throws Exception { Object obj = personClass.newInstance();//获取空参的构造函数 Method method = personClass.getDeclaredMethod("getSomeThing"); method.setAccessible(true); Object value = method.invoke(obj); System.out.println(value); } @Test public void otherMethod() throws Exception { //当前加载这个class文件的那个类加载器对象 System.out.println(personClass.getClassLoader()); //获取某个类实现的所有接口 Class[] interfaces = personClass.getInterfaces(); for (Class class1 : interfaces) { System.out.println(class1); } //反射当前这个类的直接父类 System.out.println(personClass.getGenericSuperclass()); /** * getResourceAsStream这个方法可以获取到一个输入流,这个输入流会关联到name所表示的那个文件上。 */ //path 不以’/'开头时默认是从此类所在的包下取资源,以’/'开头则是从ClassPath根下获取。其只是通过path构造一个绝对路径,最终还是由ClassLoader获取资源。 System.out.println(personClass.getResourceAsStream("/log4j.properties")); System.out.println(personClass.getResourceAsStream("log4j.properties")); //判断当前的Class对象表示是否是数组 System.out.println(personClass.isArray()); System.out.println(new String[3].getClass().isArray()); //判断当前的Class对象表示是否是枚举类 System.out.println(personClass.isEnum()); System.out.println(Class.forName("cn.itcast_04_reflect.City").isEnum()); //判断当前的Class对象表示是否是接口 System.out.println(personClass.isInterface()); System.out.println(Class.forName("cn.itcast_04_reflect.TestInterface").isInterface()); } }
2.1.动态代理
动态代理:在不修改原业务的基础上,基于原业务方法,进行重新的扩展,实现新的业务
例如下面的例子:
1、 旧业务
买家调用action,购买衣服,衣服在数据库的标价为50元,购买流程就是简单的调用。
2、 新业务
在原先的价格上可以使用优惠券,但是这个功能在以前没有实现过,我们通过代理类,代理了原先的接口方法,在这个方法的基础上,修改了返回值。
代理实现流程:
1、 书写代理类和代理方法,在代理方法中实现代理Proxy.newProxyInstance
2、 代理中需要的参数分别为:被代理的类的类加载器soneObjectclass.getClassLoader(),被代理类的所有实现接口
new Class[] { Interface.class },句柄方法new InvocationHandler()
3、 在句柄方法中复写invoke方法,invoke方法的输入有3个参数Object proxy(代理类对象), Method method(被代理类的方法),
Object[] args(被代理类方法的传入参数),在这个方法中,我们可以定制化的开发新的业务。
4、 获取代理类,强转成被代理的接口
5、 最后,我们可以像没被代理一样,调用接口的认可方法,方法被调用后,方法名和参数列表将被传入代理类的invoke方法中,进
行新业务的逻辑流程。
代理类
public static IBoss getProxyBoss(final int discountCoupon) throws Exception { Object proxedObj = Proxy.newProxyInstance(Boss.class.getClassLoader(), new Class[] { IBoss.class }, new InvocationHandler() { public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { Integer returnValue = (Integer) method.invoke(new Boss(), args);// 调用原始对象以后返回的值 return returnValue - discountCoupon; } }); return (IBoss)proxedObj; } }
新业务调用
public class ProxySaleAction { public void saleByProxy() throws Exception { IBoss boss = ProxyBoss.getProxyBoss(20);// 将代理的方法实例化成接口 System.out.println("代理经营!"); int money = boss.yifu("xxl");// 调用接口的方法,实际上调用方式没有变 System.out.println("衣服成交价:" + money); } }