java 非常好用的反射框架Reflections介紹
Reflections通過掃描classpath,索引元數(shù)據(jù),并且允許在運(yùn)行時(shí)查詢這些元數(shù)據(jù)。
使用Reflections可以很輕松的獲取以下元數(shù)據(jù)信息:1)獲取某個(gè)類型的所有子類;比如,有一個(gè)父類是TestInterface,可以獲取到TestInterface的所有子類。
2)獲取某個(gè)注解的所有類型/字段變量,支持注解參數(shù)匹配。
3)使用正則表達(dá)式獲取所有匹配的資源文件
4)獲取特定簽名方法。
通常的用法有:
引入依賴jar<dependency> <groupId>org.reflections</groupId> <artifactId>reflections</artifactId> <version>0.9.10</version></dependency>項(xiàng)目中使用:
// 初始化工具類Reflections reflections = new Reflections(new ConfigurationBuilder().forPackages(basePackages).addScanners(new SubTypesScanner()).addScanners(new FieldAnnotationsScanner())); // 獲取某個(gè)包下類型注解對(duì)應(yīng)的類Set<Class<?>> typeClass = reflections.getTypesAnnotatedWith(RpcInterface.class, true); // 獲取子類Set<Class<? extends SomeType>> subTypes = reflections.getSubTypesOf(SomeType.class); // 獲取注解對(duì)應(yīng)的方法Set<Method> resources =reflections.getMethodsAnnotatedWith(SomeAnnotation.class); // 獲取注解對(duì)應(yīng)的字段Set<Field> ids = reflections.getFieldsAnnotatedWith(javax.persistence.Id.class); // 獲取特定參數(shù)對(duì)應(yīng)的方法Set<Method> someMethods = reflections.getMethodsMatchParams(long.class, int.class); Set<Method> voidMethods = reflections.getMethodsReturn(void.class); Set<Method> pathParamMethods =reflections.getMethodsWithAnyParamAnnotated(PathParam.class); // 獲取資源文件Set<String> properties = reflections.getResources(Pattern.compile('.*.properties'));
具體也可以參見官方文檔
補(bǔ)充:Java中的反射:框架設(shè)計(jì)的靈魂
反射:框架設(shè)計(jì)的靈魂框架:半成品軟件。可以在框架的基礎(chǔ)上進(jìn)行軟件開發(fā),簡(jiǎn)化編碼
反射:將類的各個(gè)組成部分封裝為其他對(duì)象,這就是反射機(jī)制
好處:
1.可以在程序運(yùn)行過程中,操作這些對(duì)象。
2.可以解耦,提高程序的可擴(kuò)展性。
獲取Class對(duì)象的方式:1.Class.forName(“全類名”):將字節(jié)碼文件加載進(jìn)內(nèi)存,返回Class對(duì)象
多用于配置文件,將類名定義在配置文件中。讀取文件,加載類
2.類名.class:通過類名的屬性class獲取
多用于參數(shù)的傳遞
3.對(duì)象.getClass():getClass()方法在Object類中定義著。
多用于對(duì)象的獲取字節(jié)碼的方式
結(jié)論:同一個(gè)字節(jié)碼文件(*.class)在一次程序運(yùn)行過程中,只會(huì)被加載一次,不論通過哪一種方式獲取的Class對(duì)象都是同一個(gè)。
例如我們有一個(gè)Person類
public class Person { private String name; private int age; public Person(){ } public Person(String name, int age) {this.name = name;this.age = age; } public String getName() {return name; } public void setName(String name) {this.name = name; } public int getAge() {return age; } public void setAge(int age) {this.age = age; } @Override public String toString() {return 'Person{' +'name=’' + name + ’’’ +', age=' + age +’}’; }}
我們寫一個(gè)Demo用三種方式來獲取Class對(duì)象
public class Demo1 { public static void main(String [] args) throws Exception {//1、Class.forName('類名')Class cls1 = Class.forName('man.Person');System.out.println(cls1);//2、類名。classClass cls2= Person.class;System.out.println(cls2);//3、對(duì)象.getClass()Person p = new Person();Class cls3=p.getClass();System.out.println(cls3); }}Class對(duì)象功能:
獲取功能:
1.獲取成員變量們Field[] getFields() :獲取所有public修飾的成員變量
Field getField(String name) 獲取指定名稱的 public修飾的成員變量
Field[] getDeclaredFields() 獲取所有的成員變量,不考慮修飾符
Field getDeclaredField(String name)2.獲取構(gòu)造方法們
Constructor<?>[] getConstructors()Constructor getConstructor(類<?>… parameterTypes)Constructor getDeclaredConstructor(類<?>… parameterTypes)Constructor<?>[] getDeclaredConstructors()3.獲取成員方法們:
Method[] getMethods()Method getMethod(String name, 類<?>… parameterTypes)Method[] getDeclaredMethods()Method getDeclaredMethod(String name, 類<?>… parameterTypes)4.獲取全類名
String getName() * Field:成員變量
操作:
5.設(shè)置值void set(Object obj, Object value)6.獲取值
get(Object obj)7.忽略訪問權(quán)限修飾符的安全檢查
setAccessible(true):暴力反射
Constructor:構(gòu)造方法
創(chuàng)建對(duì)象:
T newInstance(Object… initargs)
如果使用空參數(shù)構(gòu)造方法創(chuàng)建對(duì)象,操作可以簡(jiǎn)化:Class對(duì)象的newInstance方法* Method:方法對(duì)象
執(zhí)行方法:
Object invoke(Object obj, Object… args)
獲取方法名稱:
String getName:獲取方法名
同樣對(duì)于上面的Person類我們對(duì)其新增帶參數(shù)和不帶參數(shù)的sleep方法并且寫一個(gè)Demo來獲取這些成員變量,構(gòu)造方法以及成員方法
Person.java
public class Person { private String name; private int age; public String a; protected String b; String c; private String d; public Person(){ } public String getName() {return name; } public void setName(String name) {this.name = name; } public int getAge() {return age; } public void setAge(int age) {this.age = age; } public Person(String name, int age) {this.name = name;this.age = age; } public void eat(){System.out.println('eat...'); } public void eat(String food){System.out.println('eat...'+food); } @Override public String toString() {return 'Person{' +'name=’' + name + ’’’ +', age=' + age +', a=’' + a + ’’’ +', b=’' + b + ’’’ +', c=’' + c + ’’’ +', d=’' + d + ’’’ +’}’; }}
Demo2.java代碼如下:
public class Demo2 { public static void main(String [] args) throws Exception {Class<Person> personClass = Person.class;//獲取成員變量Field[] fields = personClass.getFields();for(Field field:fields) { System.out.println(field);}System.out.println('--------------');Field a = personClass.getField('a');//獲取a的值Person p = new Person();Object value=a.get(p);System.out.println(value);a.set(p, 'zhangsan');//設(shè)置a的值System.out.println(p);System.out.println('=============');//獲取所有成員變量Field[] declaredFields = personClass.getDeclaredFields();for (Field declaredField : declaredFields) { System.out.println(declaredField);}Field d = personClass.getDeclaredField('d');d.setAccessible(true);//暴力反射Object value2 = d.get(p);System.out.println(value2);//獲取構(gòu)造方法Constructor<Person> constructor = personClass.getConstructor(String.class, int.class);System.out.println(constructor);Object person = constructor.newInstance('張三', 23);System.out.println(person);System.out.println('=======');Constructor<Person> constructor1 = personClass.getConstructor();System.out.println(constructor1);Object person1 = constructor1.newInstance();System.out.println(person1);//獲取成員方法Method eat_method = personClass.getMethod('eat');Person p1 = new Person();eat_method.invoke(p1);Method eat_method1 = personClass.getMethod('eat',String.class);eat_method1.invoke(p1,'飯');System.out.println('---------');Method[] methods = personClass.getMethods();for (Method method : methods) { System.out.println(method);} }}
運(yùn)行結(jié)果如下:
public java.lang.String man.Person.a-------------- null Person{name=‘null’, age=0, a=‘zhangsan’, b=‘null’, c=‘null’, d=‘null’}============= private java.lang.String man.Person.name private int man.Person.age public java.lang.String man.Person.a protectedjava.lang.String man.Person.b java.lang.String man.Person.c privatejava.lang.String man.Person.d null publicman.Person(java.lang.String,int) Person{name=‘張三’, age=23, a=‘null’,b=‘null’, c=‘null’, d=‘null’}======= public man.Person() Person{name=‘null’, age=0, a=‘null’, b=‘null’, c=‘null’, d=‘null’} eat… eat…飯--------- public java.lang.String man.Person.toString() public java.lang.String man.Person.getName() public voidman.Person.setName(java.lang.String) public voidman.Person.eat(java.lang.String) public void man.Person.eat() publicvoid man.Person.setAge(int) public int man.Person.getAge() publicfinal void java.lang.Object.wait() throwsjava.lang.InterruptedException public final voidjava.lang.Object.wait(long,int) throws java.lang.InterruptedExceptionpublic final native void java.lang.Object.wait(long) throwsjava.lang.InterruptedException public booleanjava.lang.Object.equals(java.lang.Object) public native intjava.lang.Object.hashCode() public final native java.lang.Classjava.lang.Object.getClass() public final native voidjava.lang.Object.notify() public final native voidjava.lang.Object.notifyAll()案例
需求:寫一個(gè)'框架',不能改變?cè)擃惖娜魏未a的前提下,可以幫我們創(chuàng)建任意類的對(duì)象,并且執(zhí)行其中任意方法
實(shí)現(xiàn):1. 配置文件
2. 反射
步驟:1. 將需要?jiǎng)?chuàng)建的對(duì)象的全類名和需要執(zhí)行的方法定義在配置文件中
2. 在程序中加載讀取配置文件
3. 使用反射技術(shù)來加載類文件進(jìn)內(nèi)存
4. 創(chuàng)建對(duì)象
5. 執(zhí)行方法
為了實(shí)現(xiàn)創(chuàng)建任意類的對(duì)象,并且執(zhí)行其中任意方法,我們?cè)僭蠵erson.java文件基礎(chǔ)上新增Student.java,代碼如下:
public class Student { public void sleep(){System.out.println('sleep...'); }}
那么我們需要在src目錄下添加pro.properties文件并寫入以下配置信息
className=man.StudentmethodName=sleep
接著我們來寫這個(gè)案例ReflectTest.java,代碼如下
public class ReflectTest { public static void main(String [] args) throws Exception {//加載配置文件Properties pro = new Properties();ClassLoader classLoader = ReflectTest.class.getClassLoader();InputStream is = classLoader.getResourceAsStream('pro.properties');pro.load(is);//獲取配置文件中定義的數(shù)據(jù)String className = pro.getProperty('className');String methodName = pro.getProperty('methodName');//加載該類進(jìn)內(nèi)存Class cls = Class.forName(className);Object obj = cls.newInstance();Method method = cls.getMethod(methodName);method.invoke(obj); }}
這樣我們只需改變配置文件中的信息而不需要去改變?nèi)魏未a就可以實(shí)現(xiàn)類以及類中的方法,整個(gè)目錄結(jié)構(gòu)如下
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持好吧啦網(wǎng)。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。
相關(guān)文章:
1. asp在iis7報(bào)錯(cuò)行號(hào)不準(zhǔn)問題的解決方法2. 三個(gè)不常見的 HTML5 實(shí)用新特性簡(jiǎn)介3. 淺談CSS不規(guī)則邊框的生成方案4. CSS可以做的幾個(gè)令你嘆為觀止的實(shí)例分享5. 5個(gè)HTML5的常用本地存儲(chǔ)方式詳解與介紹6. msxml3.dll 錯(cuò)誤 800c0019 系統(tǒng)錯(cuò)誤:-2146697191解決方法7. html中的form不提交(排除)某些input 原創(chuàng)8. 詳解盒子端CSS動(dòng)畫性能提升9. ASP中解決“對(duì)象關(guān)閉時(shí),不允許操作?!钡脑幃悊栴}……10. CSS百分比padding制作圖片自適應(yīng)布局
