欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java反射技術(shù)詳解及實(shí)例解析

 更新時(shí)間:2020年07月14日 14:21:00   作者:黃林晴  
這篇文章主要介紹了Java反射技術(shù)詳解及實(shí)例解析,反射可以說(shuō)是Java中最強(qiáng)大的技術(shù)了,它可以做的事情太多太多,很多優(yōu)秀的開(kāi)源框架都是通過(guò)反射完成的。如果對(duì)JAVA感興趣來(lái)可以學(xué)習(xí)一下

前言

  相信很多人都知道反射可以說(shuō)是Java中最強(qiáng)大的技術(shù)了,它可以做的事情太多太多,很多優(yōu)秀的開(kāi)源框架都是通過(guò)反射完成的,比如最初的很多注解框架,后來(lái)因?yàn)閖ava反射影響性能,所以被運(yùn)行時(shí)注解APT替代了,java反射有個(gè)開(kāi)源框架jOOR相信很多人都用過(guò),不過(guò)我們還是要學(xué)習(xí)反射的基礎(chǔ)語(yǔ)法,這樣才能自己寫(xiě)出優(yōu)秀的框架,當(dāng)然這里所講的反射技術(shù),是學(xué)習(xí)Android插件化技術(shù)、Hook技術(shù)等必不可少的!

一、基本反射技術(shù)

      1.1 根據(jù)一個(gè)字符串得到一個(gè)類

        getClass方法

 String name = "Huanglinqing";
 Class c1 = name.getClass();
 System.out.println(c1.getName());

     打印結(jié)果如下:

    Class.forName

    比如我們獲取java.lang.String的類名 

 String name = "java.lang.String";
 Class c1 = null;
 try {
 c1 = Class.forName(name);
 System.out.println(c1.getName());
 } catch (ClassNotFoundException e) {
 }

這里也通過(guò)捕獲異常,因?yàn)槲覀儌鞯倪@個(gè)字符串可能不合法,字符串合法命名是類的命名空間和類的名稱組成

    打印結(jié)果如下:

我們還可以通過(guò)c1.getSuperclass()獲取到他的父類

   Type屬性

    基本類型都有type屬性,可以得到這個(gè)基本類型的類型,比如:

Class c1 = Boolean.TYPE;
Class c2 = Byte.TYPE;
Class c3 = Float.TYPE;
Class c4 = Double.TYPE;

停停停!這些東西有啥子用,如此雞肋! 別急,一切都是為后續(xù)做準(zhǔn)備。

二、獲取類的成員

         當(dāng)類中方法定義為私有的時(shí)候我們能調(diào)用?不能!當(dāng)變量是私有的時(shí)候我們能獲取嗎?不能!但是反射可以,比如源碼中有你需要用到的方法,但是那個(gè)方法是私有的,這個(gè)時(shí)候你就可以通過(guò)反射去執(zhí)行這個(gè)私有方法,并且獲取私有變量。

       獲取類的構(gòu)造函數(shù)

       為了便于測(cè)試,我們定義一個(gè)Test類,Test類如下:(省略get和set方法)

       Test類中我們定義是三個(gè)私有變量,生成兩個(gè)公有的含參構(gòu)造方法和一個(gè)私有的含參構(gòu)造方法以及一個(gè)公有的無(wú)參構(gòu)造方法。

public class Test {
 
 private int age;
 private String name;
 private int testint;
 
 public Test(int age) {
 this.age = age;
 }
 
 public Test(int age, String name) {
 this.age = age;
 this.name = name;
 }
 
 private Test(String name) {
 this.name = name;
 }
 
 public Test() {
 }

      下面我們通過(guò)反射獲取這些構(gòu)造方法

       獲取類的所有構(gòu)造方法

 Test test = new Test();
 Class c4 = test.getClass();
 Constructor[] constructors ;
 constructors = c4.getDeclaredConstructors();

      通過(guò)getDeclaredConstructors可以返回類的所有構(gòu)造方法,返回的是一個(gè)數(shù)組因?yàn)闃?gòu)造方法可能不止一個(gè),通過(guò)getModifiers可以得到構(gòu)造方法的類型,getParameterTypes可以得到構(gòu)造方法的所有參數(shù),返回的是一個(gè)Class數(shù)組,所以我們?nèi)绻氆@取所有構(gòu)造方法以及每個(gè)構(gòu)造方法的參數(shù)類型,可以有如下代碼:

 for (int i = 0; i < constructors.length; i++) {
 System.out.print(Modifier.toString(constructors[i].getModifiers()) + "參數(shù):");
 Class[] parametertypes = constructors[i].getParameterTypes();
 for (int j = 0; j < parametertypes.length; j++) {
 System.out.print(parametertypes[j].getName() + " ");
 }
 System.out.println("");
 }

    運(yùn)行結(jié)果如下所示:

    

    這樣我們就得到了類中所有構(gòu)造方法和構(gòu)造方法中的參數(shù),那么我們?nèi)绾潍@取特定的構(gòu)造方法呢?

    獲取類中特定的構(gòu)造方法

    我們可以通過(guò)getConstructors方法獲取類中 所有的public類型的構(gòu)造方法,代碼和上面一樣就不演示了。

    我們可以通過(guò)getDeclaredConstructor()方法傳參獲取特定參數(shù)類型的構(gòu)造方法,這里注意是getDeclaredConstructor()不是  getDeclaredConstructors() ,所以返回的是一個(gè)Class對(duì)象而不是一個(gè)Class數(shù)組。

    獲取無(wú)參構(gòu)造方法直接不傳參數(shù),如下所示:

 try {
 constructors = c4.getDeclaredConstructor();
 System.out.print(Modifier.toString(constructors.getModifiers()) + );
 } catch (NoSuchMethodException e) {
 e.printStackTrace();
 }

    這里要進(jìn)行異常捕獲,因?yàn)榭赡懿淮嬖趯?duì)應(yīng)的構(gòu)造方法,打印結(jié)果如下:  

 

 如果我們想獲取有兩個(gè)參數(shù)分別為int和String類型的構(gòu)造方法,代碼如下:

 Class[] p = {int.class,String.class};
 try {
 constructors = c4.getDeclaredConstructor(p);
 System.out.print(Modifier.toString(constructors.getModifiers()) + "參數(shù):");
 Class[] parametertypes = constructors.getParameterTypes();
 for (int j = 0; j < parametertypes.length; j++) {
 System.out.print(parametertypes[j].getName() + " ");
 }
 } catch (NoSuchMethodException e) {
 e.printStackTrace();
 }

  這里我們同樣打印出構(gòu)造方法的參數(shù):

 

  調(diào)用構(gòu)造方法

   從這里開(kāi)始慢慢到了關(guān)鍵的一步,得到類的實(shí)例,我們主要借助于newInstance方法,為了方便演示我們將測(cè)試類的兩個(gè)構(gòu)造方法打印出來(lái). 

 public Test(int age, String name) {
 this.age = age;
 this.name = name;
 System.out.println("hello" + name + "i am" + age);
 }
 
 
 private Test(String name) {
 this.name = name;
 System.out.println("My Name is" +
 name);
 }

   我們先來(lái)調(diào)用public的方法,如下所示:

 Class[] p = {int.class,String.class};
 constructors = c4.getDeclaredConstructor(p);
 constructors.newInstance(24,"HuangLinqing");

 運(yùn)行打印結(jié)果如下:

  

 那么調(diào)用私有構(gòu)造方法呢,和上面一樣,只是我們要設(shè)置constructors.setAccessible(true);代碼如下:

 Class[] p = {String.class};
 constructors = c4.getDeclaredConstructor(p);
 constructors.setAccessible(true);
 constructors.newInstance("HuangLinqing");

打印結(jié)果如下:


調(diào)用類的私有方法

  如何調(diào)用類中的私有方法呢,我們先在測(cè)試類中編寫(xiě)一個(gè)測(cè)試的私有方法 如下:

 private void welcome(String tips){
 System.out.println(tips);
 }

  我們知道如果我們要正常的調(diào)用類的方法都是通過(guò)類.方法調(diào)用,所以我們調(diào)用私有方法也需要得到類的實(shí)例,而我們上面newInstace已經(jīng)得到了類的實(shí)例,這樣就好辦了。

Class[] p4 = {String.class};
 Method method = c4.getDeclaredMethod("welcome",p4);
 method.setAccessible(true);

   我們首先通過(guò) getDeclaredMethod方法獲取到這個(gè)私有方法,第一個(gè)參數(shù)是方法名,第二個(gè)參數(shù)是參數(shù)類型

   然后通過(guò)invoke方法執(zhí)行,invoke需要兩個(gè)參數(shù)一個(gè)是類的實(shí)例,一個(gè)是方法參數(shù)。    

Class[] p4 = {String.class};
 Method method = c4.getDeclaredMethod("welcome",p4);
 method.setAccessible(true);
 Object arg1s[] = {"歡迎關(guān)注代碼男人技術(shù)公眾號(hào)"};
 method.invoke(test,arg1s);

     test類的實(shí)例當(dāng)不能new 獲取的時(shí)候我們也可以通過(guò)反射獲取,就是上面的newInstance方法。打印結(jié)果如下:


 獲取類的私有字段并修改值

  看到這里你可能會(huì)說(shuō),有了set方法,什么私有不私有,test.set不就可以了,但是這里要注意我們是沒(méi)有辦法得到這個(gè)類的實(shí)例的,要不然都可以得到實(shí)例就沒(méi)有反射一說(shuō)了。我們?cè)谕ㄟ^(guò)反射得到類的實(shí)例之后先獲取字段:

Field field = c4.getDeclaredField("name");
field.setAccessible(true);
field.set(o,"代碼男人");

   o是我們上面通過(guò)反射構(gòu)造方法獲取的實(shí)例,  打印field.get(o).toString()的值如下:

   

   不過(guò)要注意的是我們修改了name的值只對(duì)當(dāng)前的實(shí)例對(duì)象有效。

   Java的基本反射語(yǔ)法就是這樣了,歡迎加入技術(shù)群一起探討!

  最后反射封裝類如下:

package jnidemo.hlq.com.hookdemo;
 
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
 
/**
 * @author Huanglinqing
 * @date 2019/4/28
 */
 
public class Reflex {
 
 /**
 * 獲取無(wú)參構(gòu)造函數(shù)
 * @param className
 * @return
 */
 public static Object createObject(String className) {
 Class[] pareTyples = new Class[]{};
 Object[] pareVaules = new Object[]{};
 
 try {
 Class r = Class.forName(className);
 return createObject(r, pareTyples, pareVaules);
 } catch (ClassNotFoundException e) {
 e.printStackTrace();
 }
 
 return null;
 }
 
 /**
 * 獲取無(wú)參構(gòu)造方法
 * @param clazz
 * @return
 */
 public static Object createObject(Class clazz) {
 Class[] pareTyple = new Class[]{};
 Object[] pareVaules = new Object[]{};
 
 return createObject(clazz, pareTyple, pareVaules);
 }
 
 /**
 * 獲取一個(gè)參數(shù)的構(gòu)造函數(shù) 已知className
 *
 * @param className
 * @param pareTyple
 * @param pareVaule
 * @return
 */
 public static Object createObject(String className, Class pareTyple, Object pareVaule) {
 
 Class[] pareTyples = new Class[]{pareTyple};
 Object[] pareVaules = new Object[]{pareVaule};
 
 try {
 Class r = Class.forName(className);
 return createObject(r, pareTyples, pareVaules);
 } catch (ClassNotFoundException e) {
 e.printStackTrace();
 }
 
 return null;
 }
 
 
 /**
 * 獲取單個(gè)參數(shù)的構(gòu)造方法 已知類
 *
 * @param clazz
 * @param pareTyple
 * @param pareVaule
 * @return
 */
 public static Object createObject(Class clazz, Class pareTyple, Object pareVaule) {
 Class[] pareTyples = new Class[]{pareTyple};
 Object[] pareVaules = new Object[]{pareVaule};
 
 return createObject(clazz, pareTyples, pareVaules);
 }
 
 /**
 * 獲取多個(gè)參數(shù)的構(gòu)造方法 已知className
 * @param className
 * @param pareTyples
 * @param pareVaules
 * @return
 */
 public static Object createObject(String className, Class[] pareTyples, Object[] pareVaules) {
 try {
 Class r = Class.forName(className);
 return createObject(r, pareTyples, pareVaules);
 } catch (ClassNotFoundException e) {
 e.printStackTrace();
 }
 
 return null;
 }
 
 
 /**
 * 獲取構(gòu)造方法
 *
 * @param clazz
 * @param pareTyples
 * @param pareVaules
 * @return
 */
 public static Object createObject(Class clazz, Class[] pareTyples, Object[] pareVaules) {
 try {
 Constructor ctor = clazz.getDeclaredConstructor(pareTyples);
 ctor.setAccessible(true);
 return ctor.newInstance(pareVaules);
 } catch (Exception e) {
 e.printStackTrace();
 }
 
 return null;
 }
 
 
 /**
 * 獲取多個(gè)參數(shù)的方法
 * @param obj
 * @param methodName
 * @param pareTyples
 * @param pareVaules
 * @return
 */
 public static Object invokeInstanceMethod(Object obj, String methodName, Class[] pareTyples, Object[] pareVaules) {
 if (obj == null) {
 return null;
 }
 
 try {
 //調(diào)用一個(gè)private方法 //在指定類中獲取指定的方法
 Method method = obj.getClass().getDeclaredMethod(methodName, pareTyples);
 method.setAccessible(true);
 return method.invoke(obj, pareVaules);
 
 } catch (Exception e) {
 e.printStackTrace();
 }
 
 return null;
 }
 
 /**
 * 獲取一個(gè)參數(shù)的方法
 * @param obj
 * @param methodName
 * @param pareTyple
 * @param pareVaule
 * @return
 */
 public static Object invokeInstanceMethod(Object obj, String methodName, Class pareTyple, Object pareVaule) {
 Class[] pareTyples = {pareTyple};
 Object[] pareVaules = {pareVaule};
 
 return invokeInstanceMethod(obj, methodName, pareTyples, pareVaules);
 }
 
 /**
 * 獲取無(wú)參方法
 * @param obj
 * @param methodName
 * @return
 */
 public static Object invokeInstanceMethod(Object obj, String methodName) {
 Class[] pareTyples = new Class[]{};
 Object[] pareVaules = new Object[]{};
 
 return invokeInstanceMethod(obj, methodName, pareTyples, pareVaules);
 }
 
 
 /**
 * 無(wú)參靜態(tài)方法
 * @param className
 * @param method_name
 * @return
 */
 public static Object invokeStaticMethod(String className, String method_name) {
 Class[] pareTyples = new Class[]{};
 Object[] pareVaules = new Object[]{};
 
 return invokeStaticMethod(className, method_name, pareTyples, pareVaules);
 }
 
 /**
 * 獲取一個(gè)參數(shù)的靜態(tài)方法
 * @param className
 * @param method_name
 * @param pareTyple
 * @param pareVaule
 * @return
 */
 public static Object invokeStaticMethod(String className, String method_name, Class pareTyple, Object pareVaule) {
 Class[] pareTyples = new Class[]{pareTyple};
 Object[] pareVaules = new Object[]{pareVaule};
 
 return invokeStaticMethod(className, method_name, pareTyples, pareVaules);
 }
 
 /**
 * 獲取多個(gè)參數(shù)的靜態(tài)方法
 * @param className
 * @param method_name
 * @param pareTyples
 * @param pareVaules
 * @return
 */
 public static Object invokeStaticMethod(String className, String method_name, Class[] pareTyples, Object[] pareVaules) {
 try {
 Class obj_class = Class.forName(className);
 return invokeStaticMethod(obj_class, method_name, pareTyples, pareVaules);
 } catch (Exception e) {
 e.printStackTrace();
 }
 
 return null;
 }
 
 /**
 * 無(wú)參靜態(tài)方法
 * @param method_name
 * @return
 */
 public static Object invokeStaticMethod(Class clazz, String method_name) {
 Class[] pareTyples = new Class[]{};
 Object[] pareVaules = new Object[]{};
 
 return invokeStaticMethod(clazz, method_name, pareTyples, pareVaules);
 }
 
 /**
 * 一個(gè)參數(shù)靜態(tài)方法
 * @param clazz
 * @param method_name
 * @param classType
 * @param pareVaule
 * @return
 */
 public static Object invokeStaticMethod(Class clazz, String method_name, Class classType, Object pareVaule) {
 Class[] classTypes = new Class[]{classType};
 Object[] pareVaules = new Object[]{pareVaule};
 
 return invokeStaticMethod(clazz, method_name, classTypes, pareVaules);
 }
 
 /**
 * 多個(gè)參數(shù)的靜態(tài)方法
 * @param clazz
 * @param method_name
 * @param pareTyples
 * @param pareVaules
 * @return
 */
 public static Object invokeStaticMethod(Class clazz, String method_name, Class[] pareTyples, Object[] pareVaules) {
 try {
 Method method = clazz.getDeclaredMethod(method_name, pareTyples);
 method.setAccessible(true);
 return method.invoke(null, pareVaules);
 } catch (Exception e) {
 e.printStackTrace();
 }
 
 return null;
 }
 
 
 public static Object getFieldObject(String className, Object obj, String filedName) {
 try {
 Class obj_class = Class.forName(className);
 return getFieldObject(obj_class, obj, filedName);
 } catch (ClassNotFoundException e) {
 e.printStackTrace();
 }
 return null;
 }
 
 public static Object getFieldObject(Class clazz, Object obj, String filedName) {
 try {
 Field field = clazz.getDeclaredField(filedName);
 field.setAccessible(true);
 return field.get(obj);
 } catch (Exception e) {
 e.printStackTrace();
 }
 
 return null;
 }
 
 
 public static void setFieldObject(Class clazz, Object obj, String filedName, Object filedVaule) {
 try {
 Field field = clazz.getDeclaredField(filedName);
 field.setAccessible(true);
 field.set(obj, filedVaule);
 } catch (Exception e) {
 e.printStackTrace();
 }
 }
 
 public static void setFieldObject(String className, Object obj, String filedName, Object filedVaule) {
 try {
 Class obj_class = Class.forName(className);
 setFieldObject(obj_class, obj, filedName, filedVaule);
 } catch (ClassNotFoundException e) {
 e.printStackTrace();
 }
 }
 
 
 public static Object getStaticFieldObject(String className, String filedName) {
 return getFieldObject(className, null, filedName);
 }
 
 public static Object getStaticFieldObject(Class clazz, String filedName) {
 return getFieldObject(clazz, null, filedName);
 }
 
 public static void setStaticFieldObject(String classname, String filedName, Object filedVaule) {
 setFieldObject(classname, null, filedName, filedVaule);
 }
 
 public static void setStaticFieldObject(Class clazz, String filedName, Object filedVaule) {
 setFieldObject(clazz, null, filedName, filedVaule);
 }
}

到此這篇關(guān)于Java反射技術(shù)詳解及實(shí)例解析的文章就介紹到這了,更多相關(guān)Java反射技術(shù)示例詳解內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • JDBC連接MySql數(shù)據(jù)庫(kù)步驟 以及查詢、插入、刪除、更新等

    JDBC連接MySql數(shù)據(jù)庫(kù)步驟 以及查詢、插入、刪除、更新等

    這篇文章主要介紹了JDBC連接MySql數(shù)據(jù)庫(kù)步驟,以及查詢、插入、刪除、更新等十一個(gè)處理數(shù)據(jù)庫(kù)信息的功能,需要的朋友可以參考下
    2018-05-05
  • Java SE使用數(shù)組實(shí)現(xiàn)高速數(shù)字轉(zhuǎn)換功能

    Java SE使用數(shù)組實(shí)現(xiàn)高速數(shù)字轉(zhuǎn)換功能

    隨著大數(shù)據(jù)時(shí)代的到來(lái),數(shù)字轉(zhuǎn)換功能變得越來(lái)越重要,在Java開(kāi)發(fā)中,數(shù)字轉(zhuǎn)換功能也是經(jīng)常用到的,下面我們就來(lái)學(xué)習(xí)一下如何使用Java SE數(shù)組實(shí)現(xiàn)高速的數(shù)字轉(zhuǎn)換功能吧
    2023-11-11
  • spring @Validated 注解開(kāi)發(fā)中使用group分組校驗(yàn)的實(shí)現(xiàn)

    spring @Validated 注解開(kāi)發(fā)中使用group分組校驗(yàn)的實(shí)現(xiàn)

    這篇文章主要介紹了spring @Validated 注解開(kāi)發(fā)中使用group分組校驗(yàn)的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-05-05
  • 關(guān)于StringUtils.isBlank()的使用及說(shuō)明

    關(guān)于StringUtils.isBlank()的使用及說(shuō)明

    這篇文章主要介紹了關(guān)于StringUtils.isBlank()的使用及說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-05-05
  • 在Mybatis中association標(biāo)簽多層嵌套的問(wèn)題

    在Mybatis中association標(biāo)簽多層嵌套的問(wèn)題

    這篇文章主要介紹了在Mybatis中association標(biāo)簽多層嵌套的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-03-03
  • SpringBoot啟動(dòng)異常Exception in thread “main“ java.lang.UnsupportedClassVersionError

    SpringBoot啟動(dòng)異常Exception in thread “main“ 

    本文主要介紹了SpringBoot啟動(dòng)異常Exception in thread “main“ java.lang.UnsupportedClassVersionError,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-07-07
  • Hadoop 中 HBase Shell命令的詳解

    Hadoop 中 HBase Shell命令的詳解

    這篇文章主要介紹了Hadoop 中 HBase Shell命令的詳解的相關(guān)資料,需要的朋友可以參考下
    2017-10-10
  • Java自定義注解用法實(shí)例小結(jié)

    Java自定義注解用法實(shí)例小結(jié)

    這篇文章主要介紹了Java自定義注解用法,結(jié)合實(shí)例形式總結(jié)分析了java常見(jiàn)的自定義注解類型、功能、用法及操作注意事項(xiàng),需要的朋友可以參考下
    2019-09-09
  • 基于@JsonSerialize和@JsonInclude注解使用方法

    基于@JsonSerialize和@JsonInclude注解使用方法

    這篇文章主要介紹了@JsonSerialize和@JsonInclude注解使用方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-10-10
  • java 關(guān)鍵字super詳解及用法

    java 關(guān)鍵字super詳解及用法

    這篇文章主要介紹了java 關(guān)鍵字super詳解及用法的相關(guān)資料,需要的朋友可以參考下
    2017-03-03

最新評(píng)論