關于Java反射機制的一個實例
JSP的規范中,有個表達式語言(Expression Language, 簡稱EL),可以算是一個微型的語言,其中對request, page, session, application中預存的JavaBean對象的引用方式很是簡單。最近正好需要寫一個支持簡單EL的taglib,所以就研究了下Java反射機制,目前基本上實現了多級bean的屬性的訪問,經測試,還是可以用的。如:
- public static void main(String[] args){
- UserBean bean = new UserBean();
- bean.setName("John Abruzzi");
- bean.setNick("Abruzzi");
- bean.setAge(24);
- AddressBean addr = new AddressBean();
- addr.setZip("0086");
- addr.setStreet("Bell Street #12");
- bean.setAddress(addr);
- System.out.println(BeanParser.doParse(bean, "bean.address.street"));
- System.out.println(BeanParser.doParse(bean, "bean.address.zip"));
- System.out.println(BeanParser.doParse(bean, "bean.name"));
- System.out.println(BeanParser.doParse(bean, "bean.nick"));
- System.out.println(BeanParser.doParse(bean, "bean.age"));
- }
需要可以輸出:
- Bell Street #12
- 0086
- John Abruzzi
- Abruzzi
- 24
反射,即由一個抽象的對象(如Object),取出這個具體對象的屬性或者方法(就EL中關于Bean的引用來說,這個定義已經夠了)。在EL中,對一個Bean的某字段進行引用,只需 ${bean.field},當然,這個bean是已經被set到容器中的,這就是Java反射機制。
我們從容器中取出以bean為名字的Object,通過Java反射機制知道它的真實類型,然后通過field以javabean規范拼出方法名,進行調用,如果這個表達式是多級的,如${bean.field.field},其中第一個field本身就是一個bean對象,同樣需要遞歸的進行解析。
大概原理就是這些了,看代碼吧:
現有一個UserBean, 其中的一個字段Address本身又是一個AddressBean。
- package elparser;
- public class AddressBean {
- private String street;
- private String zip;
- public String getZip() {
- return zip;
- }
- public void setZip(String zip) {
- this.zip = zip;
- }
- public String getStreet() {
- return street;
- }
- public void setStreet(String street) {
- this.street = street;
- }
- }
然后是UserBean
- package elparser;
- public class UserBean {
- private String name;
- private String nick;
- private AddressBean address;
- private int age;
- public int getAge(){
- return this.age;
- }
- public void setAge(int age){
- this.age = age;
- }
- public String getName() {
- return name;
- }
- public void setName(String name) {
- this.name = name;
- }
- public String getNick() {
- return nick;
- }
- public void setNick(String nick) {
- this.nick = nick;
- }
- public AddressBean getAddress() {
- return address;
- }
- public void setAddress(AddressBean address) {
- this.address = address;
- }
- }
Bean都是很簡單的,考慮到對基本類型的支持,所以在UserBean中加入一個int型的字段age
好了,看看怎么通過一個串和一個對象來取出其中的字段來:
- package elparser;
- import java.lang.reflect.Method;
- public class BeanParser {
- private static String getMethodName(String property, String prefix){
- String prop = Character.toUpperCase(property.charAt(0))+property.substring(1);
- String methodName = prefix + prop;
- return methodName;
- }
- private static Object parse(Object bean, String expr){
- Class beanClass = bean.getClass();
- Method method = null;
- Object result = null;
- try{
- //這兩步是關鍵,get方法不需要傳入參數,所以只是new出兩個空數組傳入
- method = beanClass.getMethod(getMethodName(expr, "get"), new Class[]{});
- result = method.invoke(bean, new Object[]{});
- }catch(Exception e){
- System.out.println(e.getMessage());
- }
- return result;
- }
- public static Object doParse(Object bean, String expr){
- String keys[] = expr.split("\\.");
- Object obj = null;
- for(int i = 1; i < keys.length;i++){
- obj = parse(bean, keys[i]);
- bean = obj;
- }//遞歸parse
- return obj;
- }
- public static void main(String[] args){
- UserBean bean = new UserBean();
- bean.setName("John Abruzzi");
- bean.setNick("Abruzzi");
- bean.setAge(24);
- AddressBean addr = new AddressBean();
- addr.setZip("0086");
- addr.setStreet("Bell Street #12");
- bean.setAddress(addr);
- System.out.println(BeanParser.doParse(bean, "bean.address.street"));
- System.out.println(BeanParser.doParse(bean, "bean.address.zip"));
- System.out.println(BeanParser.doParse(bean, "bean.name"));
- System.out.println(BeanParser.doParse(bean, "bean.nick"));
- System.out.println(BeanParser.doParse(bean, "bean.age"));
- }
- }
代碼比較簡短,重要部分有注釋,應該很容易理解。當然這篇文章主要是關于Java的反射機制,如果需要對EL完全支持,可以使用JavaCC做一個簡單的分析器(Apache的commons中包含一個el的項目,就是用javacc寫的分析器)。
【編輯推薦】