一、XML
1 什么是XML
XML全称为Extensible Markup Language,意思是可扩展的标记语言。XML语法上和HTML比较相似,但HTML中的元素是固定的,而XML的标签是可以由用户自定义的
2、应用常见
<1>配置文件
<2>存放数据
3、XML语法
<1>XML文档声明
①文档声明必须为<?xml开头,以?>结束;
②文档声明必须从文档的0行0列位置开始
③文档声明只有三个属性
a)version:指定XML文档版本。必须属性,因为我们不会选择1.1,只会选择1.0
b)encoding:指定当前文档度的编码。可选属性,默认值是utf-8
<2>元素element
①元素是XML文档中最重要的组成部分
②普通元素的结构开始标签、元素体、结束标签组成。例如:<hello>大家好</hello>
③元素体:元素体可以是元素,也可以是文本,例如:<b><a>你好</a></b>
④空元素:空元素只有开始标签,而没有结束标签,但元素必须自己闭合,例如:<c/>
⑤元素命令:
a)区分大小写
b)不能使用空格,不能使用冒号;
c)不建议以XML、xml、Xml开头
⑥格式化良好的XML文档,必须只有一个根元素
<3>属性
①属性是元素的一部分,它必须出现在元素的开始标签中
②属性的定义格式:属性名=属性值,其中属性值必须使用单引或双引
③一个元素可以有0~N个属性,但一个元素中不能出现同名属性
④属性名不能使用空格、冒号等特殊字符,且必须以字母开头
<4>注释
XML的注释与HTML相同,即以“<!--”开始,以“-->”结束。注释内容会被XML解析器忽略
<5>转义字符
XML中的转义字符与HTML一样
因为很多符号已经被XML文档结构所使用,所以在元素体或属性值中想使用这些符号就必须使用转义字符,例如:“<”、“>”、“&”
<6>CDATA区
当大量的转义字符出现在xml文件中时,会使xml文档的可读性大幅度降低。这时如果使用CDATA段就会好一些
在CDATA段中出现的“<”、“>”、“&”,都无需使用转义字符,这可以提高xml文档的可读性
在CDATA段中不能包含“]]>”,即CDATA段的结束定界符
4、DTD约束
常见的xml约束:DTD、Schema
<1>什么是DTD
DTD(Document Type Definition),文档类型定义,用来约束XML文档。规定XML文档中元素的名称,子元素的名称及顺序,元素的属性等
<2>DTD重点
开发中,我们很少自己编写DTD约束文档,通常情况我们都是通过框架提供的DTD约束文档,编写对应的XML文档。常见框架使用DTD约束有:struts2、hibernate等
通过提供的DTD“web-app_2_3.dtd”编写XML
<3>案列实现
①步骤1:创建web.xml,并将“web-app_2_3.dtd”拷贝相同目录下
②步骤2:从DTD文档开始处,拷贝需要的“文档声明”
③步骤3:完成xml内容编写
<4>DTD语法
①文档申明
内部DTD,在XML文档内部嵌入DTD,只对当前XML有效
外部DTD-本地DTD,DTD文档在本地系统上,公司内部自己项目使用
外部DTD-公共DTD,DTD文档在网络上,一般都有框架提供
②元素声明
实例
③属性声明
实例
5、Schema约束
<1>什么是Schema
Schema是新的XML文档约束
Schema要比DTD强大很多,是DTD替代者
Schema本身也是XML文档,但Schema文档的扩展名为xsd,而不是xml
Schema功能更强大,数据类型更完善
Schema支持名称空间
<2>Schema重点要求
与DTD一样,要求可以通过Schema约束文档编写xml文档。常见框架使用Schema的有:Spring等
通过提供“web-app_2_5.xsd”编写xml文档
案例文档中同一个“命令空间”分别使用“默认命令空间”和“显示命令空间”进行引入,所以文档中<schema>和<xsd:schema>作用一样
<3>案例实现
①步骤1:创建web.xml,并将“web-app_2_5.xsd”拷贝到同级目录
②步骤2:从xsd文档中拷贝需要的“命令空间”
③完成xml内容编写
<4>命令空间(语法)
①什么是命令空间
如果一个XML文档中使用多个Schema文件,而这些Schema文件定义了相同名称的元素时就会出现名字冲突,这就像一个java文件中使用了import java.util.*和import java.sql.*时,在使用Date类时,那么就不明确Date是哪个包下的Date了
名称空间就是用来处理元素和属性的名称冲突问题,与java中的包是同一用途。如果每个元素和属性都有自己的名称空间,那么就不会出现名字冲突问题,就像是每个类都有自己所在的包一样,那么类名就不会出现冲突
②约束文档和XML关系
当W3C提出Schema约束规范时,就提供“官方约束文档”。我们通过官方文档,必须“自定义schema约束文档”,开发中“自定义文档”由框架编写者提供,我们提供“自定义文档”限定,编写出自己的xml文档
③声明命令空间
实例:web-app_2_5.xsd
实例:web.xml
④其他介绍
自定义约束:web-app_2_5.xsd
xml文档:web.xml
综合案例:
综合案例目的是:一个“命令空间”必须在schemaLocation中有一个成对描述信息(名称 路径)
6、dom4j解析
<1>XML解析概述
当将数据存储在XML后,我们就希望通过程序获得XML的内容。如果我们使用java基础所学习的IO知识是可以完成的,不过需要非常繁琐的操作才可以完成,且开发中会遇到不同的问题(只读、读写)。人们为不同问题提供不同的解析方式,并提交对应的解析器,方便开发人员操作XML
<2>解析方式和解析器
①开发中比较常见的解析方式有三种,如下
Ⅰ、DOM:要求解析器把整个XML文档装载到内存,并解析成一个Document对象
a)优点:元素与元素之间保留结构关系,故可以进行增删改查操作
b)缺点:XML文档过大,可能出现内存溢出现象
Ⅱ、SAX:是一种速度更快,更有效的方法。它逐行扫描文档,一边扫描一边解析。并以事件驱动的方式 进行具体解析,每执行一行,都将触发对应的事件
a)优点:处理速度快,可以处理大文件
b)缺点:只能读,逐行后将释放资源
Ⅲ、PULL:Android内置的XML解析方式,类似SAX
②解析器:就是根据不同的解析方式提供的具体实现。有的解析器操作过于繁琐,为了方便开发人员,有提供易于操作的解析开发包
③常见的解析开发包:
a)JAXP:sun公司提供支持DOM和SAX开发包
b)JDom:dom4j兄弟
c)jsoup:一种处理HTML特定解析开发包
d)dom4j:比较常用的解析开发包,hibernate底层采用
<3>DOM解析原理及结构模型
XML DOM和HTML DOM类似,XML DOM将整个XML文档加载到内存,生成一个DOM树,并获得一个Document对象,通过Document对象就可以对DOM进行操作
DOM中的核心概念就是节点,在XML文档中的元素、属性、文本等,在DOM中都是节点
<4>API使用
如果需要使用dom4j,必须导入jar包
dom4j必须使用核心类SaxReader加载xml文档获得Document,通过Document对象获得文档的根元素,然后就可以操作了
常用API如下:
①SaxReader对象
a)read(...)加载执行xml文档
②Document对象
a)getRootElement()获得根元素
③Element对象
a) elements(...) 获得指定名称的所有子元素。可以不指定名称
b) element(...) 获得指定名称第一个子元素。可以不指定名称
c) getName() 获得当前元素的元素名
d) attributeValue(...) 获得指定属性名的属性值
e) elementText(...) 获得指定名称子元素的文本值
f) getText() 获得当前元素的文本内容
二、反射
1、什么是反射
<1>JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一 个对象,都能够调用它的任意一个方法和属性
<2>使用反射,可以在运行时对类Class、构造方法Constructor、普通方法Method、字段Field进行操作
2、Class对象
<1>Class对象,是对class文件(字节码文件)的描述对象
<2>获得Class对象
①已知类的完整路径名(字符串):Class.forName(...)
②已知类型:String.class,确定参数列表
③已知对象:obj.getClass()
<3>常用方法
①使用默认构造方法创建实例:newInstance()
3、Constructor对象
<1>Constructor对象,是构造方法的描述对象
<2>获得构造方法
①公共的构造方法:Constructor<T> getConstructor(Class<?>...parameterTypes),可变参数用 于确定形式参数列表
②已经声明的构造方法:
Constructor<T> getDeclaredConstructor(Class<?>...parameterTypes),获得私有的构造
<3>实例化对象实例
①newInstance(Object...initargs),可变参数用于确定实际参数列表
4、Method对象
<1>Method对象,是普通方法的描述对象
<2>获得方法:
①获得公共方法:Method getMethod(String name,Class<?>...parameterTypes)通过方法name获得 方法,可变参数为方法的形参参数列表
②获得声明方法:Method getDeclaredMethod(String name,Class<?>...parameterType)方法操 作;
<3>执行指定方法:
①Object invoke(Object obj,Object...args)执行指定对象obj,指定方法,可变参数为方法的实 际参数列表
5、Field对象
<1>Field对象,是字段的描述对象
<2>获得方法
①所有字段:Field getField(String name),通过字段名称
②声明字段:Field getDeclaredField(String name)
<3>操作
①获得内容:Object get(Object obj)
②设置内容:void set(Object obj,Object value),确定实例对象
5、案例实现
<1>准备工作
为了模拟服务器程序,且可以同时存在多个类似程序。故提供接口,接口中有3个方法,我们人为约定三个方法的调用顺序
再为接口编写实现类
测试,创建实现类实例对象
<2>反射执行
测试程序我们直接new HelloServlet,这种编程方式我们称为硬编码,即代码写死了。为了后期程序的可扩展,开发中通常使用实现类的全限定类名(cn.itcast.e_web.HelloMyServlet),通过反射加载字符串指定的类,并通过反射创建实例
<3>解析xml
使用反射我们已经可以创建对象的实例,此时我们使用的全限定类名,但程序是仍写死了,我们将其配置到xml文档中。
xml文档内容
解析实现
<4>模拟浏览器路径
上面我们已经解析xml,不过我们获得内容是固定的。我们希望如果用户访问的路径是/hello,将执行cn.itcast_web.HelloMyServlet程序,如果访问是/hello2,将执行cn.itcast.e_web.HelloMyServlet2程序。
在执行测试程序前(@Before),解析xml文件,将解析的结果存放在Map中,map中数据的格式为:
路径=实现类
解析xml思路:先解析<servlet>,将结果存放map,name=class,然后再解析<servlet-mapping>
通过name获得class,再将url=class存放到map,最后将name=class移除
模拟浏览器请求路径,通过url从map获得class,并使用反射执行实现类