一、java中的反射
1.通過反射加載類的屬性和方法實例代碼:
/** * java.lang.Class 是反射的源頭* 我們創建了一個類,通過編譯(javac.exe)生成對應的class文件,之後我們通過java.exe加載(jvm的類加載器加載)此class文件* 此class文件加載到內存後,就是一個運行時類,存在緩存區,這個運行時類本事就是一個Class的實例* 每一個運行時類只加載一次, */ Class<StudentExam> clazz = StudentExam.class; StudentExam studentExam = clazz.newInstance(); System.err.println(studentExam); System.out.println(clazz); // Field field = clazz.getField("id"); // 通過屬性調用運行時類的指定屬性:屬性是public類型Field field = clazz.getDeclaredField("id"); // 屬性是非public 類型Field[] fields = clazz.getDeclaredFields(); // 獲取運行時類本身(父類不行)所有聲明的屬性,父類使用clazz.getFields(); for (Field field2 : fields) { int i = field2.getModifiers(); String type = Modifier.toString(i);// 獲取字段屬性的數據類型System.out.println(type); } field.setAccessible(true); field.set(studentExam, 11); System.err.println(studentExam.getId()); // 通過反射調用運行時類的指定方法Method method = clazz.getMethod("setId", Integer.class); method.invoke(studentExam, 123); // 調用運行時類的指定方法Method[] methods = clazz.getMethods(); // 獲取所有運行時類及其父類中所有聲明為public的方法Method[] methods2 = clazz.getDeclaredMethods();// 獲取運行時類本身類中聲明的方法for (Method method2 : methods) { System.out.println(method2.getName()); } // * 通過對象的getClass()方法獲取對象的運行時類, Exam exam = new Exam(); Class clazzExam = exam.getClass();2.類加載器ClassLoader
/** * Description:類加載器,加載xx.properties文件,並讀取數據* @param * @author xiazhongwei * @data 2016年9月29日:下午5:32:56 * @return */ public void classLoader() throws IOException { //方法一、從當前工程下加載ClassLoader loader = this.getClass().getClassLoader(); // 路徑是包下寫:com//able//onlineExam//resources//config.properties InputStream inStream = loader.getResourceAsStream("config.properties"); // 方法二、從指定的路徑下加載文件// FileInputStream fileInputStream = new FileInputStream(new File("config.properties")); Properties properties = new Properties(); properties.load(inStream); // properties.load(fileInputStream); String prop = properties.getProperty("domain"); System.out.println(prop); }3.動態代理
靜態代理:代理類和目標對象的類型都是在編譯期間確定下來,不利於程序的擴展。同時每個代理類只能為一個接口服務,這樣一來程序開發中必然產生過多的代理。
動態代理:客戶通過代理類來調用其他對象的方法,並且是在程序運行時,根據需要動態創建目標類的代理對象。
代理設計模式的原理:
使用一個代理將對象包裝起來,然後用該代理對象取代原始對象,任何對原始對象的調用都要通過代理,代理對象決定的那個是否以及何時將方法調用
package com.test.junit; import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy; public class ProxyTest { public static void main(String[] args) { RealSubject realSubject = new RealSubject(); MyInvocationHandler myInvocationHandler = new MyInvocationHandler(); Object object = myInvocationHandler.bind(realSubject); Subject subject = (Subject) object; subject.action(); }}// 動態代理的使用interface Subject{ void action();}// 被代理類class RealSubject implements Subject{ @Override public void action() { System.out.println("我是被代理類,記得執行我哦。。。。"); } } class MyInvocationHandler implements InvocationHandler{ Object object;// 實現了接口的被代理類的對象的聲明/** * Description:①給被代理的對象實例化②返回一個代理類對象* @param * @author xiazhongwei * @data 2016年9月29日:下午4:13:43 * @return */ public Object bind(Object object){ this.object = object; return Proxy.newProxyInstance(object.getClass().getClassLoader(), object.getClass().getInterfaces(), this); } /** * 當通過代理類的對象發起對被重寫的方法的調用時,都會轉化為對如下的invok方法的調用*/ @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { Object returnObject = method.invoke(object, args); return returnObject; }}4.動態代理與AOP
示例一、
package com.atguigu.spring.aop; public interface ArithmeticCalculator { int add(int i, int j); int sub(int i, int j); int mul(int i, int j); int div(int i, int j);} package com.atguigu.spring.aop; import org.springframework.stereotype.Component; @Component("arithmeticCalculator")public class ArithmeticCalculatorImpl implements ArithmeticCalculator { @Override public int add(int i, int j) { int result = i + j; return result; } @Override public int sub(int i, int j) { int result = i - j; return result; } @Override public int mul(int i, int j) { int result = i * j; return result; } @Override public int div(int i, int j) { int result = i / j; return result; } } package com.atguigu.spring.aop; public class ArithmeticCalculatorLoggingImpl implements ArithmeticCalculator { @Override public int add(int i, int j) { System.out.println("The method add begins with [" + i + "," + j + "]"); int result = i + j; System.out.println("The method add ends with " + result); return result; } @Override public int sub(int i, int j) { System.out.println("The method sub begins with [" + i + "," + j + "]"); int result = i - j; System.out.println("The method sub ends with " + result); return result; } @Override public int mul(int i, int j) { System.out.println("The method mul begins with [" + i + "," + j + "]"); int result = i * j; System.out.println("The method mul ends with " + result); return result; } @Override public int div(int i, int j) { System.out.println("The method div begins with [" + i + "," + j + "]"); int result = i / j; System.out.println("The method div ends with " + result); return result; } } package com.atguigu.spring.aop; import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy;import java.util.Arrays; public class ArithmeticCalculatorLoggingProxy { //要代理的對象private ArithmeticCalculator target; public ArithmeticCalculatorLoggingProxy(ArithmeticCalculator target) { super(); this.target = target; } //返回代理對象public ArithmeticCalculator getLoggingProxy(){ ArithmeticCalculator proxy = null; // 代理對像有哪一個類加載器負責加載ClassLoader loader = target.getClass().getClassLoader(); // 代理對象的類型,即其中有哪些方法Class [] interfaces = new Class[]{ArithmeticCalculator.class}; // 當調用代理對象的其中方法時,執行下面的代碼InvocationHandler h = new InvocationHandler() { /** * proxy: 代理對象。 一般不使用該對象* method: 正在被調用的方法* args: 調用方法傳入的參數*/ @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { // 在方法內部不會直接調用proxy對象的某個方法,proxy.toString()會造成死循環調用invoke方法String methodName = method.getName(); //打印日誌System.out.println("[before] The method " + methodName + " begins with " + Arrays.asList(args)); //調用目標方法Object result = null; try { //前置通知result = method.invoke(target, args); //返回通知, 可以訪問到方法的返回值} catch (NullPointerException e) { e.printStackTrace(); //異常通知, 可以訪問到方法出現的異常} //後置通知. 因為方法可以能會出異常, 所以訪問不到方法的返回值//打印日誌System.out.println("[after] The method ends with " + result); return result; } }; /** * loader: 代理對象使用的類加載器。 * interfaces: 指定代理對象的類型. 即代理代理對像中可以有哪些方法. * h: 當具體調用代理對象的方法時, 應該如何進行響應, 實際上就是調用InvocationHandler 的invoke 方法*/ proxy = (ArithmeticCalculator) Proxy.newProxyInstance(loader, interfaces, h); return proxy; }} package com.atguigu.spring.aop; import org.springframework.context.ApplicationContext;import org.springframework.context.support.ClassPathXmlApplicationContext; public class Main { public static void main(String[] args) { // ArithmeticCalculator arithmeticCalculator = new ArithmeticCalculatorImpl(); ArithmeticCalculator arithmeticCalculator = new ArithmeticCalculatorLoggingImpl(); arithmeticCalculator = new ArithmeticCalculatorLoggingProxy(arithmeticCalculator).getLoggingProxy(); int result = arithmeticCalculator.add(11, 12); System.out.println("result:" + result); result = arithmeticCalculator.div(21, 3); System.out.println("result:" + result); } }示例二、
package com.test.junit; import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy; public class ProxyTest { public static void main(String[] args) { RealSubject realSubject = new RealSubject(); MyInvocationHandler myInvocationHandler = new MyInvocationHandler(); Object object = myInvocationHandler.bind(realSubject); Subject subject = (Subject) object; subject.action(); }}// 動態代理的使用interface Subject{ void action();}// 被代理類class RealSubject implements Subject{ @Override public void action() { System.out.println("我是被代理類,記得執行我哦。。。。"); } } class MyInvocationHandler implements InvocationHandler{ Object object;// 實現了接口的被代理類的對象的聲明/** * Description:①給被代理的對象實例化②返回一個代理類對象* @param * @author xiazhongwei * @data 2016年9月29日:下午4:13:43 * @return */ public Object bind(Object object){ this.object = object; return Proxy.newProxyInstance(object.getClass().getClassLoader(), object.getClass().getInterfaces(), this); } /** * 當通過代理類的對象發起對被重寫的方法的調用時,都會轉化為對如下的invok方法的調用*/ @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { Object returnObject = method.invoke(object, args); return returnObject; }}感謝閱讀此文,希望能幫助到大家,謝謝大家對本站的支持!