下面通過實(shí)例代碼給大家介紹Tomcat解析XML和反射創(chuàng)建對(duì)象原理,具體代碼如下所示:
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.List;
import org.dom4j.Document;
import org.dom4j.DocumentException;
import org.dom4j.Element;
import org.dom4j.io.SAXReader;
public class ServerReadXML1 {
public static void main(String[] args)
throws DocumentException, ClassNotFoundException, InstantiationException, IllegalAccessException,
NoSuchMethodException, SecurityException, IllegalArgumentException, InvocationTargetException {
// 現(xiàn)在假如在瀏覽器中輸入一個(gè)Servlet的url-pattern
String urlPattern = "/first";
// 根據(jù)urlPattern 獲取類名
String className = getClassByUrl(urlPattern);
// 根據(jù)全類名獲取Class對(duì)象
Class clazz = Class.forName(className);
// 通過反射clazz對(duì)象創(chuàng)建指定對(duì)象
Object obj = clazz.newInstance();
// 獲取service方法
Method method = clazz.getDeclaredMethod("service");
// 獲取權(quán)限
method.setAccessible(true);
// 執(zhí)行service方法
method.invoke(obj);
}
private static String getClassByUrl(String urlPattern) throws DocumentException {
// 1.創(chuàng)建SAXReader對(duì)象
SAXReader reader = new SAXReader();
// 2.讀取文件
Document document = reader.read(ServerReadXML1.class.getClassLoader().getResourceAsStream("web.xml"));
// 3.獲取根節(jié)點(diǎn)
Element rootElement = document.getRootElement();
//System.out.println(rootElement.getName());
// 4.獲取根節(jié)點(diǎn)下 的子節(jié)點(diǎn)
List<Element> servletList = rootElement.elements();
// 記錄與urlPattern相同的servlet-name標(biāo)簽的內(nèi)容
String servletName = "";
// 記錄servlet標(biāo)簽中的servlet-class的內(nèi)容
// servletClassName的內(nèi)容也就是Servlet的全類名
String servletClassName = "";
// 5.遍歷子節(jié)點(diǎn)
for (Element servletElement : servletList) {
//System.out.println(servletElement.getName());
// 判斷如果是servlet-mapping標(biāo)簽時(shí),執(zhí)行代碼
if ("servlet-mapping".equals(servletElement.getName())) {
// 獲取url-pattern標(biāo)簽對(duì)象
Element url = servletElement.element("url-pattern");
// 判斷標(biāo)簽的內(nèi)容和入的urlPattern值是否相同
if (urlPattern.equals(url.getText())) {
// 記錄與urlPattern相同的servlet-name標(biāo)簽的內(nèi)容
// 如果相同,則記錄ServletName
// 獲取servlet-mapping中的servelt-name的內(nèi)容
servletName = servletElement.element("servlet-name").getText();
}
}
}
// 再次遍歷
for (Element servletElement : servletList) {
// 判斷如果是servlet標(biāo)簽時(shí),執(zhí)行此代碼
if ("servlet".equals(servletElement.getName())) {
// 判斷上一次的遍歷獲取的servletName的值和這次遍歷中的servlet-name的內(nèi)容是否相同
if (servletName.equals(servletElement.element("servlet-name").getText())) {
// 如果相同記錄servletClassName
servletClassName = servletElement.element("servlet-class").getText();
}
}
}
// 返回Servlet的全類名 servletClassName
return servletClassName;
}
}
1.反射的獲取Class 4種方式
@Test
public void test1() throws ClassNotFoundException {
//1.類名.class
Class clazz = String.class;
System.out.println(clazz);
//2.對(duì)象.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對(duì)象 可以獲取其內(nèi)部的屬性
Class clazz = Class.forName("com.atguigu.bean.User");
User user = new User();
//Field對(duì)象 代表中類的屬性 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.反射使用方法的常用方法
@Test
public void test3() throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, SecurityException, IllegalArgumentException, InvocationTargetException {
Class clazz = Class.forName("com.atguigu.bean.User");
//通過反射創(chuàng)建對(duì)象
Object obj = clazz.newInstance();
//現(xiàn)在想要設(shè)置name值
String fileName = "name";
//創(chuàng)建一個(gè)方法名
String methodName = "set" + fileName.substring(0, 1).toUpperCase() //N
+ fileName.substring(1).toLowerCase(); //ame
//根據(jù)方法名 獲取公共方法
Method method = clazz.getMethod(methodName, String.class);
//執(zhí)行指定的方法
method.invoke(obj, "yangjian");
System.out.println(obj);
}
總結(jié)
以上所述是小編給大家介紹的Tomcat解析XML和反射創(chuàng)建對(duì)象原理,希望對(duì)大家有所幫助,如果大家有任何疑問請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
如果你覺得本文對(duì)你有幫助,歡迎轉(zhuǎn)載,煩請(qǐng)注明出處,謝謝!