Tomcat解析XML和反射建立對象原理java
1 import java.lang.reflect.InvocationTargetException; 2 import java.lang.reflect.Method; 3 import java.util.List; 4
5 import org.dom4j.Document; 6 import org.dom4j.DocumentException; 7 import org.dom4j.Element; 8 import org.dom4j.io.SAXReader; 9
10 public class ServerReadXML1 { 11
12 public static void main(String[] args) 13 throws DocumentException, ClassNotFoundException, InstantiationException, IllegalAccessException, 14 NoSuchMethodException, SecurityException, IllegalArgumentException, InvocationTargetException { 15
16 // 如今假如在瀏覽器中輸入一個Servlet的url-pattern
17 String urlPattern = "/first"; 18
19 // 根據urlPattern 獲取類名
20 String className = getClassByUrl(urlPattern); 21
22 // 根據全類名獲取Class對象
23 Class clazz = Class.forName(className); 24
25 // 經過反射clazz對象建立指定對象
26 Object obj = clazz.newInstance(); 27
28 // 獲取service方法
29 Method method = clazz.getDeclaredMethod("service"); 30
31 // 獲取權限
32 method.setAccessible(true); 33
34 // 執行service方法
35 method.invoke(obj); 36
37 } 38
39 private static String getClassByUrl(String urlPattern) throws DocumentException { 40
41 // 1.建立SAXReader對象
42 SAXReader reader = new SAXReader(); 43
44 // 2.讀取文件
45 Document document = reader.read(ServerReadXML1.class.getClassLoader().getResourceAsStream("web.xml")); 46
47 // 3.獲取根節點
48 Element rootElement = document.getRootElement(); 49 //System.out.println(rootElement.getName()); 50
51 // 4.獲取根節點下 的子節點
52 List<Element> servletList = rootElement.elements(); 53
54 // 記錄與urlPattern相同的servlet-name標籤的內容
55 String servletName = ""; 56
57 // 記錄servlet標籤中的servlet-class的內容 58 // servletClassName的內容也就是Servlet的全類名
59 String servletClassName = ""; 60
61 // 5.遍歷子節點
62 for (Element servletElement : servletList) { 63 //System.out.println(servletElement.getName()); 64
65 // 判斷若是是servlet-mapping標籤時,執行代碼
66 if ("servlet-mapping".equals(servletElement.getName())) { 67
68 // 獲取url-pattern標籤對象
69 Element url = servletElement.element("url-pattern"); 70
71 // 判斷標籤的內容和入的urlPattern值是否相同
72 if (urlPattern.equals(url.getText())) { 73
74 // 記錄與urlPattern相同的servlet-name標籤的內容 75 // 若是相同,則記錄ServletName 76 // 獲取servlet-mapping中的servelt-name的內容
77 servletName = servletElement.element("servlet-name").getText(); 78
79 } 80
81 } 82
83 } 84
85 // 再次遍歷
86 for (Element servletElement : servletList) { 87 // 判斷若是是servlet標籤時,執行此代碼
88 if ("servlet".equals(servletElement.getName())) { 89
90 // 判斷上一次的遍歷獲取的servletName的值和此次遍歷中的servlet-name的內容是否相同
91 if (servletName.equals(servletElement.element("servlet-name").getText())) { 92
93 // 若是相同記錄servletClassName
94 servletClassName = servletElement.element("servlet-class").getText(); 95
96 } 97
98 } 99
100 } 101
102 // 返回Servlet的全類名 servletClassName
103 return servletClassName; 104 } 105
106 }
1.反射的獲取Class 4種方式web
@Test public void test1() throws ClassNotFoundException { //1.類名.class
Class clazz = String.class; System.out.println(clazz); //2.對象.getClass()
Class clazz1 = "abc".getClass(); System.out.println(clazz1); //3.Class.forName();
Class clazz2 = Class.forName("java.lang.String"); System.out.println(clazz2); //4.ClassLoader .loadClass("全類名")
Class clazz3 = ReflectTest1.class.getClassLoader().loadClass("java.lang.String"); System.out.println(clazz3); }
2.反射使用屬性的經常使用方法瀏覽器
@Test public void test2() throws ClassNotFoundException, NoSuchFieldException, SecurityException, IllegalArgumentException, IllegalAccessException { //獲取Class對象 能夠獲取其內部的屬性
Class clazz = Class.forName("com.atguigu.bean.User"); User user = new User(); //Field對象 表明中類的屬性 getField只能獲取公共屬性
Field field = clazz.getField("email"); System.out.println(field); //此種方式破壞代碼的封裝性 不推薦使用
Field field2 = clazz.getDeclaredField("id"); System.out.println(field2); field2.setAccessible(true); field2.setInt(user, 1001); System.out.println(user); }
3.反射使用方法的經常使用方法app
@Test public void test3() throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, SecurityException, IllegalArgumentException, InvocationTargetException { Class clazz = Class.forName("com.atguigu.bean.User"); //經過反射建立對象
Object obj = clazz.newInstance(); //如今想要設置name值
String fileName = "name"; //建立一個方法名
String methodName = "set" + fileName.substring(0, 1).toUpperCase() //N
+ fileName.substring(1).toLowerCase(); //ame //根據方法名 獲取公共方法
Method method = clazz.getMethod(methodName, String.class); //執行指定的方法
method.invoke(obj, "yangjian"); System.out.println(obj); }