=波波日志 > JavaScript/Ajax > JavaScript解析XML的方法总结=

[转]JavaScript解析XML的方法总结

  最近一个项目中要用到树形列表,本来是用mztree的,但数据用了xml,本来xml就是树形目录,就打算用JavaScript直接解析xml文件了,网上找到一些JavaScript操作xml的方法,在此做一个总结。

  我的xml文件Login.xml如下:
+展开
-XML
<?xml version="1.0" encoding="utf-8" ?>
<Login>
<Character>
<C Text="热血Value="0"></C>
<C Text="弱气Value="1"></C>
<C Text="激情Value="2"></C>
<C Text="冷静Value="3"></C>
<C Text="冷酷Value="4"></C>
</Character>
<Weapon>
<W Text="光束剑Value="0"></W>
<W Text="光束配刀Value="1"></W>
</Weapon>
<EconomyProperty>
<P Text="平均型Value="0"></P>
<P Text="重视攻击Value="1"></P>
<P Text="重视敏捷Value="2"></P>
<P Text="重视防御Value="3"></P>
<P Text="重视命中Value="4"></P>
</EconomyProperty>
</Login>


 现在我需要对这个xml文件的内容进行操作。

  首先,我们需要加载这个xml文件,JavaScript中加载xml文件,是通过XMLDOM来进行的:

+展开
-JavaScript
// 加载xml文档
loadXML = function(xmlFile)
{
var xmlDoc;
if(window.ActiveXObject)
{
xmlDoc = new ActiveXObject('Microsoft.XMLDOM');
xmlDoc.async = false;
xmlDoc.load(xmlFile);
}
else if (document.implementation&&document.implementation.createDocument)
{
xmlDoc = document.implementation.createDocument(''''null);
xmlDoc.load(xmlFile);
}
else
{
return null;
}
return xmlDoc;
}


  xml文件对象出来了, 接下去我就要对这个文档进行操作了。

  比如说,我们现在需要得到节点Login/Weapon/W的第一个节点的属性,那么我们可以如下进行:

+展开
-JavaScript
// 首先对xml对象进行判断
checkXMLDocObj = function(xmlFile)
{
var xmlDoc = loadXML(xmlFile);
if(xmlDoc==null)
{

  alert('您的浏览器不支持xml文件读取,于是本页面禁止您的操作,推荐使用IE5.0以上可以解决此问题!');

window.location.href='/Index.aspx';
}
return xmlDoc;
}

  // 然后开始获取需要的Login/Weapon/W的第一个节点的属性值

var xmlDoc = checkXMLDocObj('/EBS/XML/Login.xml');
var v = xmlDoc.getElementsByTagName('Login/Weapon/W')[0].childNodes.getAttribute('Text')


  而我在我的程序中的写法是这样子的,当然我在程序中的写法是已经应用到实际中的了.一并给出来,以供查看。

+展开
-JavaScript
initializeSelect = function(oid, xPath)
{
var xmlDoc = checkXMLDocObj('/EBS/XML/Login.xml');
var n;
var l;
var e = $(oid);
if(e!=null)
{
n = xmlDoc.getElementsByTagName(xPath)[0].childNodes;
l = n.length;
for(var i=0; i{
var option = document.createElement('option');
option.value = n[i].getAttribute('Value');
option.innerHTML = n[i].getAttribute('Text');
e.appendChild(option);
}
}
}


  上面的访问代码中,我们是通过xmlDoc.getElementsByTagName(xPath)来进行的。

  还可以通过xmlDoc.documentElement.childNodes(1)..childNodes(0).getAttribute('Text')进行访问。


  一些常用方法:其实和XHTML中的js操作dom原属差不多,不过没有document.getElementById,只有node.getElementsByTagName。

  ◆ xmlDoc.documentElement.childNodes(0).nodeName,可以得到这个节点的名称;

  ◆ xmlDoc.documentElement.childNodes(0).nodeValue,可以得到这个节点的值. 这个值是来自于这样子的xml格式:<a>b</b>, 于是可以得到b这个值;

  ◆ xmlDoc.documentElement.childNodes(0).hasChild,可以判断是否有子节点。

  最好是使用getElementsByTagName(xPath)的方法对节点进行访问,因为这样子可以直接通过xPath来定位节点,这样子会有更好的性能。
类别:JavaScript/Ajax  作者:转载  日期:2009-07-02  【评论:6 阅读:】  上一篇 下一篇
 
    • 闫昌盛

  • 日期:2009-7-7 16:46:35  IP:220.201.*.*
    那我要是让xml中的数据显示在我指定的位置,比如一个表格中,显示的过程怎么写呢?
    这个是我现在遇到的一个问题,想请教一下!
    管理员回复(2009-7-8 16:22:04)
    给一个简单的例子给你
    data.xml
    +展开
    -XML
    <?xml version="1.0" encoding="gb2312"?>
    <employees>
      <employee name="J.Doe"><job>Programmer</job><salary>32768</salary></employee>
      <employee name="A.Baker"><job>Sales</job><salary>70000</salary></employee>
      <employee name="Big Cheese"><job>CEO</job><salary>1000000</salary></employee>
    </employees>


    test.html,这里使用的是设置div的innerHTML,如果要操作table,你可以查看这篇文章
    ie和firefox下操作table对象的异同

    +展开
    -HTML
    <div id="dvTB"></div><!---这里是接受分析xml文件生成table的父容器--->
    <script>

    //注意,为了考虑兼容性,请尽量使用 getElementsByTagName来获取节点集合,而不是使用 selectNode('xpath'),selectNode在w3c浏览器不支持,只有ie支持

    function ParseXML(dom){
      var dv=document.getElementById('dvTB'),innerHTML='<table border="1"><tr align="center"><td>Name</td><td>Job</td><td>Salary</td></tr>',node
         ,employee=dom.getElementsByTagName("employee");//获取所有employee标签
         
       //遍历employee
       for(var i=0;i<employee.length;i++){
         innerHTML+='<tr><td>'+employee[i].getAttribute('name')+'</td>'//使用getAttribute获取该节点的name属性,和xmhtml差不多的
         node=employee[i].getElementsByTagName("job")[0];//获取当前节点下的所有job节点,因为此节点下只有一个job,所以索引为0
         node=node.firstChild;//注意文本节点也算一个节点,所以还得使用firstChild
         innerHTML+='<td>'+node.nodeValue+'</td>';//获取job节点内的内容
         //同理获取该employee节点下的salary节点,上面为了说明,多分了几个步骤,下面就简化了
         innerHTML+='<td>'+employee[i].getElementsByTagName("salary")[0].firstChild.nodeValue+'</td>';
         innerHTML+='</tr>';//闭合tr
       }  
       innerHTML+='</table>';//闭合table
       //设置div的innerHTML属性
       dv.innerHTML=innerHTML;
    }
    function loadXML(url)
    {
     var xmldoc;
     if(document.implementation&&document.implementation.createDocument){
      xmldoc=document.implementation.createDocument("","",null);
      xmldoc.onload=function(){ParseXML(xmldoc);}
      xmldoc.load(url);
     }
     else if(window.ActiveXObject){
      xmldoc=new ActiveXObject("Microsoft.XMLDOM");
      xmldoc.onreadystatechange=function(){if(xmldoc.readyState==4)ParseXML(xmldoc);}
      xmldoc.load(url);
     }
    }

    window.onload=function(){loadXML("data.xml");}</script> 

    • 兄弟

  • 日期:2009-7-14 11:15:40  IP:118.113.*.*
    兄弟问个问题,我在用S60开发widget,我把这段代码放在IE浏览器下就好使,在S60手机模拟器就不行了。
    downLoad('xml/talk.xml', 0);
    function downLoad(dataSource, selectItem){

    XMLHttpRequestObject = new XMLHttpRequest();
    // XMLHttpRequestObject = new ActiveXObject("Microsoft.XMLHTTP");
    // XMLHttpRequestObject.overrideMimeType("text/xml")



    try {
    XMLHttpRequestObject.open("GET", dataSource);
    }
    catch (exception) {
    window.alert("resorce havent!");
    }


    XMLHttpRequestObject.onreadystatechange = function(){

    if (XMLHttpRequestObject.readyState == 4 && XMLHttpRequestObject.status == 200) {


    var xmlDocument = XMLHttpRequestObject.responseXML;

    var temp = xmlDocument.documentElement.childNodes;

    var count = temp[selectItem].childNodes.length;

    window.alert(count);
    for (i = 0; i < count / 3; i++)
    document.getElementById('meeting').innerHTML += temp[selectItem].getElementsByTagName("id")[i].childNodes[0].nodeValue + temp[selectItem].getElementsByTagName("english")[i].firstChild.data + "<br />" + temp[selectItem].getElementsByTagName("chinese")[i].firstChild.data + "<br />";


    delete XMLHttpRequestObject;
    XMLHttpRequestObject = null;
    }
    }
    XMLHttpRequestObject.send(null);

    }
    在模拟器显示count是0,在IE就好使的。兄弟帮忙解答一下谢谢了。
    管理员回复(2009-7-14 13:44:39)
    上面的代码是使用acx来创建的,使用了注册表中的microsoft.xmldom对象。
    手机不知道手机支不支持acx,而且也没有注册表吧。。。。O(∩_∩)O~

    • 哥们拜托了

  • 日期:2009-7-16 15:48:20  IP:118.113.*.*
    <?xml version="1.0" encoding="utf-8"?>

    <talk>

    <meeting>
    <setence>
    <id>1.</id>
    <english>Let me introduce you to Mr. Li, general manager of our company.</english>
    <chinese>让我介绍你认识,这是我们的总经理,李先生。</chinese>
    </setence>

    <word>
    <id>1.</id>
    <english>meeting</english>
    <chinese>会议</chinese>
    </word>
    </meeting>

    <Airport>
    <setence>
    <id>1.</id>
    <english>May I see your passport, please?</english>
    <chinese>麻烦请给我你的护照</chinese>
    </setence>

    <word>
    <id>1.</id>
    <english>Airport</english>
    <chinese>机场</chinese>
    </word>
    </Airport>
    这是一个XML文件
    我想动态在html生成树怎么搞,搞成这样
    -meeting
    setence
    word
    -airport
    setence
    word
    然后setence和word可以点击的,如果点击meeting的setence,就把所有的数据都显示出来
    谢谢了。
    管理员回复(2009-7-17 10:41:11)
    你的xml文件talk节点没闭合,帮你写了一个简单的结构。

    data.xml
    +展开
    -XML
    <?xml version="1.0" encoding="utf-8"?>
    <talk>
    <meeting>
    <setence>
    <id>1.</id>
    <english>Let me introduce you to Mr. Li, general manager of our company.</english>
    <chinese>让我介绍你认识,这是我们的总经理,李先生。</chinese> 
    </setence>

    <word>
    <id>1.</id>
    <english>meeting</english>
    <chinese>会议</chinese> 
    </word> 
    </meeting>

    <Airport>
    <setence>
    <id>1.</id>
    <english>May I see your passport, please?</english>
    <chinese>麻烦请给我你的护照</chinese>
    </setence>

    <word>
    <id>1.</id>
    <english>Airport</english>
    <chinese>机场</chinese>
    </word>
    </Airport>
    </talk>


    test.html
    +展开
    -HTML

    <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">>
    <html xmlns="http://www.w3.org/1999/xhtml">
    <head runat="server">
        <meta http-equiv="Content-Type" content="text/html;charset=utf-8" />
        <title>xml解析</title>
    </head>
    <body>
    <div id="dvResult"></div>
    <script type="text/javascript">
    //如果你修改了结构,注意修改此函数的内容隐藏显示的代码
    function dispCT(o){var n=o.nextSibling;n.style.display=n.style.display=='none'?'block':'none';}
    function $s(node,tagName){return node.getElementsByTagName(tagName);}//=======简化  getElementByTagName

    function parseCT(node){//分析内容的函数
     return  '  Id:'+$s(node,"id")[0].firstChild.nodeValue
             +'<br/>  english:'+$s(node,"english")[0].firstChild.nodeValue
             +'<br/>  chinese:'+$s(node,"chinese")[0].firstChild.nodeValue;
    }

    function parseXML(doc){
      var meeting=$s(doc,"meeting")[0]//获取meeting的数组中的第一项
         ,airport=$s(doc,"Airport")[0]//注意大小写
         ,htmlMeeting="<div>-meeting",htmlAir="<div>-airport";//用来存储解析meeting和airport的结果的变量
      //分析meeting节点,默认隐藏内容项
      var setence=$s(meeting,"setence")[0],word=$s(meeting,"word")[0];
      htmlMeeting+='<div onclick="dispCT(this)"> setence</div><div style="display:none">'+parseCT(setence)+'</div>'//setence
                  +'<div onclick="dispCT(this)"> word</div><div style="display:none">'+parseCT(word)+'</div>';//word
      //同理分析airport
      setence=$s(airport,"setence")[0];word=$s(airport,"word")[0];
      htmlAir+='<div onclick="dispCT(this)"> setence</div><div style="display:none">'+parseCT(setence)+'</div>'//setence
              +'<div onclick="dispCT(this)"> word</div><div style="display:none">'+parseCT(word)+'</div>';//word
      htmlMeeting+='</div>';
      htmlAir+='</div>';
      document.getElementById('dvResult').innerHTML=htmlMeeting+htmlAir;
    }

    function loadXML(url)
    {
     var xmldoc;
     if(document.implementation&&document.implementation.createDocument){
      xmldoc=document.implementation.createDocument("","",null);
      xmldoc.onload=function(){parseXML(xmldoc);}
      xmldoc.load(url);
     }
     else if(window.ActiveXObject){
      xmldoc=new ActiveXObject("Microsoft.XMLDOM");
      xmldoc.onreadystatechange=function(){if(xmldoc.readyState==4)parseXML(xmldoc);}
      xmldoc.load(url);
     }
    }

    window.onload=function(){loadXML("data.xml");}
    </script> 
    </body>
    </html>

    • 感谢

  • 日期:2009-7-21 10:04:29  IP:118.113.*.*
    哥们你太厉害,我想都想不出来我初学,请问这个东西怎么学才可以
    管理员回复(2009-7-21 10:56:56)
    O(∩_∩)O~多谢支持。
    其实很简单的
    第一,你的xml结构要规则些,这样使用getElementsByTagName获取到的节点集合就比较唯一。这样方便遍历节点集合。

    第二,就是要学习下js和xhtml了,xml和xhtml的结构获取一样的。只是xml的dom没有getElementById,但是xhtml有。

    至于兼容方面,尽量使用getElementByTagName来获取节点,selectNodes之类的xpath方法好像只有ie支持,ff不支持。获取节点内容使用text,nodeValue属性,节点属性值就使用getAttriute("节点名称")。

    ie下可以使用node.xml这样获取里面的xml字符串,但是ff不支持,你可以使用XMLSerializer类来序列化

    具体如何使用看下面的文章
    [原]javascript如何解析xml文件,兼容ie及ff

    • 困惑中

  • 日期:2009-9-6 16:14:51  IP:219.135.*.*
    大哥能不能加你QQ,我想请教你几个问题。唉。最近弄这个弄的头大了。
    管理员回复(2009-9-7 8:41:49)
    qq上班一般不用,不知道有msn没。

    • 困惑中

  • 日期:2009-9-6 16:15:33  IP:219.135.*.*
    我的QQ:282567546
    管理员回复(2009-9-7 8:42:12)
    有msn可以加,qq上班不能用。

发表留言
  • 昵称:
  • 头像:
  • 验证码:  
Powered by showbo
CopyRight Code-design.cn 桂ICP备05005887号