使用DOM生成XML文件

本文介绍如何使用DOM解析器生成和修改XML文件。通过解析XML文件创建Document对象,然后使用Transformer对象将其转换为新的XML文件。同时展示了如何利用DOM API添加、修改和删除节点。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
6.11   使用DOM生成XML文件
解析器通过在内存中建立和XML结构相对应的树状结构数据,使得应用程序可以方便地获得XML文件中的数据。JAXP也提供了使用内存中的树状结构数据建立一个XML文件的API,即使用解析器得到的Document对象建立一个新的XML文件。
1 .Transformer对象
我们已经知道,解析器的parse方法将整个被解析的XML文件封装成一个Document节点返回,我们可以对Document节点进行修改,然后使用 Transformer对象将一个Document节点变换为一个XML文件。
即使解析器不调用parse方法,也可以得到一个Document节点。解析器通过调用newDocument()可得到一个Document节点,例如:
Document  document= builder.newDocument();
应用程序可通过修改这样的Document节点,然后使用 Transformer对象将一个Document节点变换为一个XML文件。
使用 Transformer对象将一个Document节点变换为一个XML文件需要经过如下  步骤。
使用javax.xml.transform包中的TransformerFactory类建立一个对象:
TransformerFactory transFactory=TransformerFactory. newInstance()
步骤 中得到的transFactory对象调用newTransformer()方法得到一个Transformer对象:
Transformer transformer=transFactory. newTransformer();
Transformer类在javax.xml.transform包中。
将被变换的Document对象封装到一个DOMSource对象中:
DOMSource  domSource= new  DOMSource(document);
DOMSource类在javax.xml.transform.dom包中。
将变换得到XML文件对象封装到一个StreamResult对象中:
File file= new  File( "newXML.xml" );
FileOutputStream out= new  FileOutputStream(file);
StreamResult xmlResult= new  StreamResult(out);
StreamResult类在javax.xml.transform.stream包中。
最后,Transformer 对象transformer 调用transform方法实施变换:
transformer.transform(domSource, xmlResult);
2 .用于修改Document的常用方法
Node接口是Document的父接口,提供了许多用来修改、增加和删除节点的方法:
Node appendChild(Node newChild) 节点调用该方法可以向当前节点增加一个新的子节点,并返回这个新节点。
Node removeChild(Node oldChild)  throws  DOMException节点调用该方法删除参数指定的子节点,并返回被删除的子节点。
Node replaceChild(Node newChild, Node oldChild) 节点调用该方法可以替换子节点,并返回被替换的子节点。
Element接口本身除了从Node接口继承的方法外,也提供了用来增加节点的方法:
Attr removeAttributeNode(Attr oldAttr) 删除Element节点的属性。
void  setAttribute(String name, String value)为Element节点增加新的属性及属性值,如果该属性已经存在,新的属性将替换旧的属性。
Text接口本身除了从Node接口继承的方法外,也提供了用来修改节点内容的方法:
Text replaceWholeText(String content) 替换当前Text节点的文本内容。
void  appendData(String arg) 向当前Text节点尾加文本内容。
void  insertData( int  offset, String arg) 向当前Text节点插入文本内容,插入的位置由参数offset指定,即第offset个字符的后继位置。
void  deleteData( int  offset, int  count) 删除当前节点的文本内容中的一部分。被删除的范围由参数offset和count指定,即从第offset个字符后续的count个字符。
void  replaceData( int  offset, int  count, String arg)将当前Text节点中文本内容的一部分替换为参数arg指定的内容,被替换的范围由参数offset和count指定,即从第offset个字符后续的count个字符。
3 .用DOM建立XML文件
在下面的例子 10 中,解析器解析一个XML文件:“cha6_10.xml”,然后修改Document对象,并用Transformer得到一个新的XML文件:“newXML.xml”。
例子 10
Cha6_10.xml
<?xml  version= "1.0"   encoding= "UTF-8"  ?>
<考试成绩单>
<高等数学>
<考生姓名>张三 </考生姓名>
<成绩>  89  </成绩>
</高等数学>
<高等数学>
<考生姓名> 李四 </考生姓名>
<成绩>  66  </成绩>
</高等数学>
</考试成绩单>
JAXPTen.java
import  javax.xml.transform.*;
import  javax.xml.transform.stream.*;
import  javax.xml.transform.dom.*;
import  org.w3c.dom.*;
import  javax.xml.parsers.*;
import  java.io.*;
public  class  JAXPTen
{
public  static  void  main(String args[])
{
ModifyNode modify= new  ModifyNode();
try  {
DocumentBuilderFactory  factory=
DocumentBuilderFactory. newInstance();
DocumentBuilder  builder= factory. newDocumentBuilder();
Document  document= builder. parse( new  File( "Cha6_10.xml" )) ;
Element root=document.getDocumentElement() ;
NodeList nodeList=root.getChildNodes();
modify.modifyNode(nodeList);
TransformerFactory transFactory=TransformerFactory. newInstance();
Transformer transformer=transFactory. newTransformer();
DOMSource  domSource= new  DOMSource(document);
File file= new  File( "newXML.xml" );
FileOutputStream out= new  FileOutputStream(file);
StreamResult xmlResult= new  StreamResult(out);
transformer.transform(domSource, xmlResult);
}         
catch (Exception e)
{
System.out.println(e);
}
}
}
class   ModifyNode
{
int  m= 0 ;
public  void  modifyNode(NodeList nodeList)
{
int  size=nodeList.getLength();
for ( int  k= 0 ;k<size;k++)
{
Node node=nodeList.item(k);
if (node.getNodeType()==Node.TEXT_NODE)
{
Text textNode=(Text)node;
int  length=textNode.getLength();
String str=textNode.getWholeText().trim();
try {
double  d=Double.parseDouble(str);
if (d>= 90 &&d<= 100 )
textNode.insertData(length, "(优dd秀)" );
else   if (d>= 80 &&d< 90 )
textNode.insertData(length, "(良好)" );
else   if (d>= 60 &&d< 80 )
textNode.insertData(length, "(及格)" );
else
textNode.insertData(length, "(不及格)" );
}
catch (NumberFormatException ee)
{}               
}
if (node.getNodeType()==Node.ELEMENT_NODE)
{
Element elementNode=(Element)node;
String name=elementNode.getNodeName();
if (elementNode.hasChildNodes())
{
elementNode.setAttribute( "考试性质" , "闭卷" ) ;
}
NodeList nodes=elementNode.getChildNodes();
modifyNode(nodes);
}
}
}
}
上述例子 10 得到的XML文件“newXML.xml”的内容如下:
newXML.xml
<?xml version= "1.0"  encoding= "UTF-8" ?><考试成绩单>
<高等数学 考试性质= "闭卷" >
<考生姓名 考试性质= "闭卷" >张三 </考生姓名>
<成绩 考试性质= "闭卷" 89  (良好)</成绩>
</高等数学>
<高等数学 考试性质= "闭卷" >
<考生姓名 考试性质= "闭卷" > 李四 </考生姓名>
<成绩 考试性质= "闭卷" 66  (及格)</成绩>
</高等数学>
</考试成绩单>
上述例子 10 中的DOM解析器利用已知的XML文件产生一个Document对象,然后对内存中的Document对象修改后,再生成一个新的XML文件。在下面的例子 11 中,DOM解析器调用newDocument()得到一个Document对象,“JavaEleven.java”产生的XML文件是“火车时刻表.xml”,用浏览器打开它的效果如图 6.12 所示。
6.12   用DOM生成的XML文件
例子 11
JAXPEleven.java
import  javax.xml.transform.*;
import  javax.xml.transform.stream.*;
import  javax.xml.transform.dom.*;
import  org.w3c.dom.*;
import  javax.xml.parsers.*;
import  java.io.*;
public  class  JAXPEleven
{
public  static  void  main(String args[])
{
try  {
String  train[]={ "T126次" , "K256次" , "L345次" },
type[] ={ "特快" , "普快" , "临客" },
startTime []={ "18:36" , "22:56" , "10:12" };
DocumentBuilderFactory  factory=
DocumentBuilderFactory. newInstance();
DocumentBuilder  builder= factory. newDocumentBuilder();
Document  document= builder.newDocument();
document.setXmlVersion( "1.0" );
Element 火车时刻表=document.createElement( "火车时刻表" );
document.appendChild(火车时刻表);
for ( int  k= 1 ;k<=train.length;k++)
{
火车时刻表.appendChild(document.createElement( "车次" ));
}
NodeList  nodeList=document.getElementsByTagName( "车次" );
int  size=nodeList.getLength();
for ( int  k= 0 ;k<size;k++)
{
Node node=nodeList.item(k);
if (node.getNodeType()==Node.ELEMENT_NODE)
{
Element elementNode=(Element)node;
elementNode.setAttribute( "类别" , type[k]);
elementNode.appendChild(document.createElement( "名字" ));
elementNode.appendChild(document.createElement( "开车时间" ));
}
}
nodeList=document.getElementsByTagName( "名字" );
size=nodeList.getLength();
for ( int  k= 0 ;k<size;k++)
{
Node node=nodeList.item(k);
if (node.getNodeType()==Node.ELEMENT_NODE)
{
Element elementNode=(Element)node;
elementNode.appendChild(document.createTextNode(train[k]));
}
}
nodeList=document.getElementsByTagName( "开车时间" );
size=nodeList.getLength();
for ( int  k= 0 ;k<size;k++)
{
Node node=nodeList.item(k);
if (node.getNodeType()==Node.ELEMENT_NODE)
{
Element elementNode=(Element)node;
elementNode.appendChild(document.createTextNode
(startTime[k]));
}
}
TransformerFactory transFactory=TransformerFactory.newInstance();
Transformer transformer=transFactory.newTransformer();
DOMSource  domSource= new  DOMSource(document);
File file= new  File( "火车时刻表.xml" );
FileOutputStream out= new  FileOutputStream(file);
StreamResult xmlResult= new  StreamResult(out);
transformer.transform(domSource, xmlResult);
}
catch (Exception e)
{
System.out.println(e);
}
}      
}

112052976.jpg
















本文转自yunlielai51CTO博客,原文链接:http://blog.51cto.com/4925054/1281625,如需转载请自行联系原作者

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值