文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Java使用SAX解析xml的示例

2024-04-02 19:55

关注

一、SAX解析xml简介

SAX是Simple API for Xml的简写,主要功能是用于对xml文档进行解析。由于该方式采用的是事件驱动(callback回调机制)解析方式,所以有速度快、占内存少的优点,当然这些优点也仅限于xml的读取操作,SAX是无法对读取的XML元素进行修改的。如果要修改节点元素则需要使用DOC方式进行将xml文件读取,它会将xml读取成document树结构对象,这样可用对节点元素进行编辑操作;DOC方式的缺点也比较明显:占内存大、解析速度较慢。

所以仅用于读取xml操作,使用SAX方式是比较好的方式。

二、SAX解析XML实例

创建一个解析的xml文件


<?xml version="1.0" encoding="utf-8"?>
<persons>
  <user>
    <userId>1001</userId>
    <userName>张三</userName>
  </user>
  <user>
    <userId>1002</userId>
    <userName>李四</userName>
  </user>
</persons>

创建一个XMLparseHandler用于自定义xml解析


public class Customhandler extends DefaultHandler2 {

  List<Map> list = new ArrayList<>();
  Map map = null;
  String tag = "";

  @Override
  public void startDocument() throws SAXException {
    System.out.println("开始解析xml");
  }

  @Override
  public void startElement(String uri, String localName, String qName, Attributes attributes) throws SAXException {
    System.out.println("开始解析元素: <"+ qName + ">");
    if(qName == "user"){
      map = new HashMap();
    }
    tag = qName;
  }

  @Override
  public void characters(char[] ch, int start, int length) throws SAXException {
    String text = new String(ch, start, length).trim();
    if(text != null && !text.isEmpty() && tag!=null&& tag!=""){
      map.put(tag, text);
      if(!map.containsKey(tag)){
      }
      System.out.println("解析到元素值:"+ text);
    }
  }

  @Override
  public void endElement(String uri, String localName, String qName) throws SAXException {
    System.out.println("结束解析元素: <"+ qName + ">");
    if(qName.equals("user")){
      list.add(map);
    }
    tag = "";
  }

  @Override
  public void endDocument() throws SAXException {
    System.out.println("结束解析xml");
  }
}

创建SAX解析对象解析xml


public static void main(String[] args) throws ParserConfigurationException, SAXException, IOException {
  //创建xml解析工厂
  SAXParserFactory factory = SAXParserFactory.newInstance();
  //创建xml解析对象
  SAXParser parser = factory.newSAXParser();
  File file = new File("test/custom/user.xml");
  InputStream inputStream = new FileInputStream(file);
  Customhandler customhandler = new Customhandler();
  //方式一
  //parser.parse(inputStream, customhandler);

	//方式二
  InputSource source = new InputSource(file.toURI().toURL().toString());
  XMLReader xmlParser = parser.getXMLReader();
  xmlParser.setContentHandler(customhandler);
  xmlParser.parse(source);
  List c = customhandler.list;
  inputStream.close();
}


//打印结果为:
开始解析xml
开始解析元素: <persons>
开始解析元素: <user>
开始解析元素: <userId>
解析到元素值:1001
结束解析元素: <userId>
开始解析元素: <userName>
解析到元素值:张三
结束解析元素: <userName>
结束解析元素: <user>
开始解析元素: <user>
开始解析元素: <userId>
解析到元素值:1002
结束解析元素: <userId>
开始解析元素: <userName>
解析到元素值:李四
结束解析元素: <userName>
结束解析元素: <user>
结束解析元素: <persons>
结束解析xml

三、SAX的实际应用

在tomcat源码中,有一个Digester对象,这个Digester是tomcat启动时,初始化各个容器(service、engine、Connetor)的执行者,而Digester执行容器初始化的依据是解析配置文件server.xml的内容,根据xml的具体配置进行来初始化容器。

下面是Digester的类的一些主要方法:


//org.apache.tomcat.util.digester.Digester#parse(org.xml.sax.InputSource)
public class Digester extends DefaultHandler2 {
  
  //读取解析xml
  public Object parse(InputSource input) throws IOException, SAXException {
    configure();
    getXMLReader().parse(input);
    return root;
  }
  
  //对每个xml标签进行解析,并执行于之对应的Rule规则列表
  public void startElement(String namespaceURI, String localName, String qName, Attributes list)
    throws SAXException {
    boolean debug = log.isDebugEnabled();
    // Parse system properties
    list = updateAttributes(list);
    // Save the body text accumulated for our surrounding element
    bodyTexts.push(bodyText);
    bodyText = new StringBuilder();
    // the actual element name is either in localName or qName, depending
    // on whether the parser is namespace aware
    String name = localName;
    if ((name == null) || (name.length() < 1)) {
      name = qName;
    }

    // Compute the current matching rule
    StringBuilder sb = new StringBuilder(match);
    if (match.length() > 0) {
      sb.append('/');
    }
    sb.append(name); //根据每次xml节点的名称拼接成匹配url
    match = sb.toString();

    // Fire "begin" events for all relevant rules(根据namespaceURI匹配获取的Rule规则列表,有顺序规则)
    List<Rule> rules = getRules().match(namespaceURI, match);
    matches.push(rules);
    if ((rules != null) && (rules.size() > 0)) {
      for (Rule value : rules) {
        try {
          Rule rule = value;
          if (debug) {
            log.debug(" Fire begin() for " + rule);
          }
          //依次执行begin方法
          rule.begin(namespaceURI, name, list);
        } catch (Exception e) {
          log.error("Begin event threw exception", e);
          throw createSAXException(e);
        } catch (Error e) {
          log.error("Begin event threw error", e);
          throw e;
        }
      }
    } else {
      if (debug) {
        log.debug(" No rules found matching '" + match + "'.");
      }
    }

  }
}

与之对应的server.xml片段如下:


<Service name="Catalina">

  <Connector port="8081" protocol="HTTP/1.1"
        connectionTimeout="20000"
        redirectPort="8443" />

  <Engine name="Catalina" defaultHost="localhost">

     <Host name="localhost" appBase="webapps"
        unpackWARs="true" autoDeploy="true">
     </Host>
  </Engine>
</Service>

Digester读取到上面这些xml标签后,就会从外向里进行嵌套解析,将这些标签创建为与之对应的java类实例,也就是tomcat的主体容器结构。

以上就是Java使用SAX解析xml的示例的详细内容,更多关于Java使用SAX解析xml的资料请关注编程网其它相关文章!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯