服务热线:13616026886

技术文档 欢迎使用技术文档,我们为你提供从新手到专业开发者的所有资源,你也可以通过它日益精进

位置:首页 > 技术文档 > JAVA > 新手入门 > 基础入门 > 查看文档

java与xml联合编程之dom篇


  dom初步
  dom是document object model的缩写,即文档对象模型。前面说过,xml将数据组织为一颗树,所以dom就是对这颗树的一个对象描叙。通俗的说,就是通过解析xml文档,为xml文档在逻辑上建立一个树模型,树的节点是一个个对象。我们通过存取这些对象就能够存取xml文档的内容。
  下面我们来看一个简单的例子,看看在dom中,我们是如何来操作一个xml文档的。
  这是一个xml文档,也是我们要操作的对象:
  
  
  good-bye serialization, hello java!
  

  下面,我们需要把这个文档的内容解析到一个个的java对象中去供程序使用,利用jaxp,我们只需几行代码就能做到这一点。首先,我们需要建立一个解析器工厂,以利用这个工厂来获得一个具体的解析器对象:
  documentbuilderfactory dbf = documentbuilderfactory.newinstance();
  我们在这里使用documentbuilderfacotry的目的是为了创建与具体解析器无关的程序,当documentbuilderfactory类的静态方法newinstance()被调用时,它根据一个系统变量来决定具体使用哪一个解析器。又因为所有的解析器都服从于jaxp所定义的接口,所以无论具体使用哪一个解析器,代码都是一样的。所以当在不同的解析器之间进行切换时,只需要更改系统变量的值,而不用更改任何代码。这就是工厂所带来的好处。这个工厂模式的具体实现,可以参看下面的类图。
  documentbuilder db = dbf.newdocumentbuilder();
  当获得一个工厂对象后,使用它的静态方法newdocumentbuilder()方法可以获得一个documentbuilder对象,这个对象代表了具体的dom解析器。但具体是哪一种解析器,微软的或者ibm的,对于程序而言并不重要。
  然后,我们就可以利用这个解析器来对xml文档进行解析了:
  document doc = db.parse("c:/xml/message.xml");
  documentbuilder的parse()方法接受一个xml文档名作为输入参数,返回一个document对象,这个document对象就代表了一个xml文档的树模型。以后所有的对xml文档的操作,都与解析器无关,直接在这个document对象上进行操作就可以了。而具体对document操作的方法,就是由dom所定义的了。
   java与xml联合编程之dom篇
  jaxp支持w3c所推荐的dom 2。如果你对dom很熟悉,那么下面的内容就很简单了:只需要按照dom的规范来进行方法调用就可以。当然,如果你对dom不清楚,也不用着急,后面我们会有详细的介绍。在这儿,你所要知道并牢记的是:dom是用来描叙xml文档中的数据的模型,引入dom的全部原因就是为了用这个模型来操作xml文档的中的数据。dom规范中定义有节点(即对象)、属性和方法,我们通过这些节点的存取来存取xml的数据。
  从上面得到的document对象开始,我们就可以开始我们的dom之旅了。使用document对象的getelementsbytagname()方法,我们可以得到一个nodelist对象,一个node对象代表了一个xml文档中的一个标签元素,而nodelist对象,观其名而知其意,所代表的是一个node对象的列表:
  nodelist nl = doc.getelementsbytagname("message");
  我们通过这样一条语句所得到的是xml文档中所有标签对应的node对象的一个列表。然后,我们可以使用nodelist对象的item()方法来得到列表中的每一个node对象:
  node my_node = nl.item(0);
  当一个node对象被建立之后,保存在xml文档中的数据就被提取出来并封装在这个node中了。在这个例子中,要提取message标签内的内容,我们通常会使用node对象的getnodevalue()方法:
  string message = my_node.getfirstchild().getnodevalue();
  请注意,这里还使用了一个getfirstchild()方法来获得message下面的第一个子node对象。虽然在message标签下面除了文本外并没有其它子标签或者属性,但是我们坚持在这里使用getfirsechild()方法,这主要和w3c对dom的定义有关。w3c把标签内的文本部分也定义成一个node,所以先要得到代表文本的那个node,我们才能够使用getnodevalue()来获取文本的内容。
  现在,既然我们已经能够从xml文件中提取出数据了,我们就可以把这些数据用在合适的地方,来构筑应用程序。
  下面的内容,我们将更多的关注dom,为dom作一个较为详细的解析,使我们使用起来更为得心应手。
  dom详解
  1.基本的dom对象
  dom的基本对象有5个:document,node,nodelist,element和attr。下面就这些对象的功能和实现的方法作一个大致的介绍。
  [[the no.5 picture.]]
  document对象代表了整个xml的文档,所有其它的node,都以一定的顺序包含在document对象之内,排列成一个树形的结构,程序员可以通过遍历这颗树来得到xml文档的所有的内容,这也是对xml文档操作的起点。我们总是先通过解析xml源文件而得到一个document对象,然后再来执行后续的操作。此外,document还包含了创建其它节点的方法,比如createattribut()用来创建一个attr对象。它所包含的主要的方法有:
  
  createattribute(string):用给定的属性名创建一个attr对象,并可在其后使用setattributenode方法来放置在某一个element对象上面。
  
  createelement(string):用给定的标签名创建一个element对象,代表xml文档中的一个标签,然后就可以在这个element对象上添加属性或进行其它的操作。
  
  createtextnode(string):用给定的字符串创建一个text对象,text对象代表了标签或者属性中所包含的纯文本字符串。如果在一个标签内没有其它的标签,那么标签内的文本所代表的text对象是这个element对象的唯一子对象。
  
  getelementsbytagname(string):返回一个nodelist对象,它包含了所有给定标签名字的标签。
  
  getdocumentelement():返回一个代表这个dom树的根节点的element对象,也就是代表xml文档根元素的那个对象。
  
  node对象是dom结构中最为基本的对象,代表了文档树中的一个抽象的节点。在实际使用的时候,很少会真正的用到node这个对象,而是用到诸如element、attr、text等node对象的子对象来操作文档。node对象为这些对象提供了一个抽象的、公共的根。虽然在node对象中定义了对其子节点进行存取的方法,但是有一些node子对象,比如text对象,它并不存在子节点,这一点是要注意的。node对象所包含的主要的方法有:
  
  appendchild(org.w3c.dom.node):为这个节点添加一个子节点,并放在所有子节点的最后,如果这个子节点已经存在,则先把它删掉再添加进去。
  
  getfirstchild():如果节点存在子节点,则返回第一个子节点,对等的,还有getlastchild()方法返回最后一个子节点。
  
  getnextsibling():返回在dom树中这个节点的下一个兄弟节点,对等的,还有getprevioussibling()方法返回其前一个兄弟节点。
  
  getnodename():根据节点的类型返回节点的名称。
  
  getnodetype():返回节点的类型。
  
  getnodevalue():返回节点的值。
  
  haschildnodes():判断是不是存在有子节点。
  
  hasattributes():判断这个节点是否存在有属性。
  
  getownerdocument():返回节点所处的document对象。
  
  insertbefore(org.w3c.dom.node new,org.w3c.dom.node ref):在给定的一个子对象前再插入一个子对象。
  removechild(org.w3c.dom.node):删除给定的子节点对象。
  
  replacechild(org.w3c.dom.node new,org.w3c.dom.node old):用一个新的node对象代替给定的子节点对象。
  
  nodelist对象,顾名思义,就是代表了一个包含了一个或者多个node的列表。可以简单的把它看成一个node的数组,我们可以通过方法来获得列表中的元素:
  
  getlength():返回列表的长度。
  
  item(int):返回指定位置的node对象。
  
  element对象代表的是xml文档中的标签元素,继承于node,亦是node的最主要的子对象。在标签中可以包含有属性,因而element对象中有存取其属性的方法,而任何node中定义的方法,也可以用在element对象上面。
  getelementsbytagname(string):返回一个nodelist对象,它包含了在这个标签中其下的子孙节点中具有给定标签名字的标签。
  
  gettagname():返回一个代表这个标签名字的字符串。
  
  getattribute(string):返回标签中给定属性名称的属性的值。在这儿需要主要的是,应为xml文档中允许有实体属性出现,而这个方法对这些实体属性并不适用。这时候需要用到getattributenodes()方法来得到一个attr对象来进行进一步的操作。
  
  getattributenode(string):返回一个代表给定属性名称的attr对象。
  
  attr对象代表了某个标签中的属性。attr继承于node,但是因为attr实际上是包含在element中的,它并不能被看作是element的子对象,因而在dom中attr并不是dom树的一部分,所以node中的getparentnode(),getprevioussibling()和getnextsibling()返回的都将是null。也就是说,attr其实是被看作包含它的element对象的一部分,它并不作为dom树中单独的一个节点出现。这一点在使用的时候要同其它的node子对象相区别。
  需要说明的是,上面所说的dom对象在dom中都是用接口定义的,在定义的时候使用的是与具体语言无关的idl语言来定义的。因而,dom其实可以在任何面向对象的语言中实现,只要它实现了dom所定义的接口和功能就可以了。同时,有些方法在dom中并没有定义,是用idl的属性来表达的,当被映射到具体的语言时,这些属性被映射

扫描关注微信公众号