赞
踩
反射的概念:
Java反射机制,是说在运行时刻,对于任何一个类,都能够知道它的所有属性和方法;对任意一个对象,都能够调用它的任意方法和属性;这种动态获取信息以及动态调用方法的功能称为java的反射机制。
假定给出以下Person类:
- 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());
-
-
- }
-
- }
动态代理:
假设我们当前类service中方法实现的业务可能不能够满足当先客户的要求,需要我们重新修改service中的方法,但是service的方法不只在我们这个模块使用,在其他模块也在调用,其他模块调用的时候,现有的service方法已经能够满足业务需求,所以我们不能只为了我们的业务而修改service,导致其他模块授影响。
那怎么办呢?
可以通过动态代理的方式,扩展我们的service中的方法实现,使得在原有的方法中增加更多的业务,而不是实际修改service中的方法,这种实现技术就叫做动态代理。
动态代理:在不修改原业务的基础上,基于原业务方法,进行重新的扩展,实现新的业务。
例如下面的例子:
买家调用action,购买衣服,衣服在数据库的标价为50元,购买流程就是简单的调用。
在原先的价格上可以使用优惠券,但是这个功能在以前没有实现过,我们通过代理类,代理了原先的接口方法,在这个方法的基础上,修改了返回值。
旧业务定义了一个接口和该接口的实现类,以及旧业务的测试实现:
- package cn.itcast_05_proxy.service;
- /**
- * 这是一个业务的接口,这个接口中的业务就是返回衣服的价格
- *
- */
- public interface IBoss {//接口
- int yifu(String size);
- }
-
-
-
- package cn.itcast_05_proxy.service.impl;
-
- import cn.itcast_05_proxy.service.IBoss;
-
-
- /**
- * 实现了卖衣服的接口
- */
- public class Boss implements IBoss{
- public int yifu(String size){
- System.err.println("天猫小强旗舰店,老板给客户发快递----衣服型号:"+size);
- //这件衣服的价钱,从数据库读取
- return 50;
- }
- }
-
-
- package cn.itcast_05_proxy.action;
-
- import org.junit.Test;
-
- import cn.itcast_05_proxy.service.IBoss;
- import cn.itcast_05_proxy.service.impl.Boss;
-
- public class SaleAction {
- /**
- * 旧业务的实现效果
- * 不使用代理,直接调用方法
- * 方法中规定什么业务,就只能调用什么业务,规定什么返回值,就只能输出什么返回值
- */
- @Test
- public void saleByBossSelf() throws Exception {
- IBoss boss = new Boss();
- System.out.println("老板自营!");
- int money = boss.yifu("xxl");// 老板自己卖衣服,不需要客服,结果就是没有聊天记录
- System.out.println("衣服成交价:" + money);
- }
- }
新业务增加促销优惠券功能,但是因为可能其他模块还需要原有功能,不能改动旧业务代码。
动态代理:先写个代理的工具类,对接口方法进行代理,将旧业务+新增功能后的逻辑封装。
- public class ProxyBoss {
- /**
- * 对接口方法进行代理
- */
- @SuppressWarnings("unchecked")
- public static <T> T getProxy(final int discountCoupon,
- final Class<?> interfaceClass, final Class<?> implementsClass)
- throws Exception {
- return (T) Proxy.newProxyInstance(interfaceClass.getClassLoader(),
- new Class[] { interfaceClass }, new InvocationHandler() {
- public Object invoke(Object proxy, Method method,
- Object[] args) throws Throwable {
- Integer returnValue = (Integer) method.invoke(
- implementsClass.newInstance(), args);// 调用原始对象以后返回的值
- return returnValue - discountCoupon;
- }
- });
- }
- }
新业务的测试实现:
- import org.junit.Test;
-
- import cn.itcast_05_proxy.proxyclass.ProxyBoss;
- import cn.itcast_05_proxy.service.IBoss;
- import cn.itcast_05_proxy.service.impl.Boss;
-
- /**
- * 什么是动态代理? 简单的写一个模板接口,剩下的个性化工作,好给动态代理来完成!
- */
- public class ProxySaleAction {
-
- /**
- *使用代理,在这个代理中,只代理了Boss的yifu方法
- *定制化业务,可以改变原接口的参数、返回值等
- */
- @Test
- public void saleByProxy() throws Exception {
- IBoss boss = ProxyBoss.getProxy(10,IBoss.class,Boss.class);// 将代理的方法实例化成接口
- //IBoss boss = new Boss();// 将代理的方法实例化成接口
- System.out.println("代理经营!");
- int money = boss.yifu("xxl");// 调用接口的方法,实际上调用方式没有变
- System.out.println("衣服成交价:" + money);
- }
- }
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。