Java StAX 解析器 - 查詢 XML 文件



Java StAX 解析器是一個 Java API,用於解析 XML 文件並查詢所需資訊。此 API 基於事件,因此我們無需載入整個 XML 文件即可查詢它。當解析器識別每個事件時,只有當客戶端程式實現該事件時才會執行相應的操作。

在本章中,我們將學習如何查詢 XML 文件以獲取所需資訊。

使用 Java StAX 解析器查詢 XML

以下是使用 Java StAX 解析器查詢 XML 文件需要遵循的步驟:

  • 步驟 1:建立 XMLInputFactory 例項
  • 步驟 2:讀取 XML
  • 步驟 3:解析 XML
  • 步驟 4:查詢元素

有關前三個步驟,請參考本章

步驟 4:查詢元素

完成前三個步驟後,我們擁有 XMLEventReader 來從 XML 文件獲取事件。透過使用各種事件並實現它們,我們可以查詢 XML 文件。讓我們詳細瞭解如何做到這一點。

按文字內容查詢元素

我們可以使用getData()方法獲取 XML 文件中任何元素的文字內容。Characters 介面的此方法以字串的形式返回當前事件的字元資料。使用此方法,我們可以查詢所有元素以查詢所需的文字內容。

示例

下面顯示的cars.xml檔案包含許多具有不同文字內容的carname元素。讓我們查詢此檔案以檢視是否存在“Bentley 2”汽車。

<?xml version = "1.0"?>
<cars>
      <carname company="Ferarri" >Ferarri 101</carname>
      <carname company="Lamborgini">Lamborgini 001</carname>
      <carname company="Lamborgini">Lamborgini 002</carname>
      <carname company="Lamborgini">Lamborgini 003</carname>
      <carname company="Bentley">Bentley 1</carname>
      <carname company="Bentley">Bentley 2</carname>
      <carname company="Bentley">Bentley 3</carname>
</cars>

以下QueryTextContent.java程式使用 FileReader 物件讀取 cars.xml 檔案。當遇到 CHARACTERS 事件型別時,getData() 方法用於獲取文字內容。如果該資料等於“Bentley 2”,那麼我們將更新 'found' 布林變數。在 END_DOCUMENT 事件中,我們將其列印到控制檯。

import java.io.FileReader;
import javax.xml.stream.XMLEventReader;
import javax.xml.stream.XMLInputFactory;
import javax.xml.stream.XMLStreamConstants;
import javax.xml.stream.events.Characters;
import javax.xml.stream.events.XMLEvent;

public class QueryTextContent {
   public static void main(String args[]) {
      try {
    	  
         //Creating XMLInputFactory instance
     	 XMLInputFactory factory = XMLInputFactory.newInstance();
     	 
     	 //Reading the XML
     	 FileReader fileReader = new FileReader("cars.xml");
     	 
     	 //Parsing the XML
         XMLEventReader eventReader =
          factory.createXMLEventReader(fileReader);
         
         //Querying the XML
         boolean found=false;
         while(eventReader.hasNext()) {
        	 XMLEvent event = eventReader.nextEvent();
        	 if(event.getEventType()==XMLStreamConstants.CHARACTERS) {
              Characters characters = event.asCharacters();
              String textContent = characters.getData();
              if(textContent.equals("Bentley 2"))
              	  found=true;
             }
        	 if(event.getEventType()==XMLStreamConstants.END_DOCUMENT) {
        		 if(found) 
        		    System.out.println("Bentley 2 car is found");
        		 else
        			System.out.println("Bentley 2 car is not found");
        	 }
         }
          
      } catch(Exception e) {
		   e.printStackTrace();
      }
   }
}

輸出

由於 cars.xml 檔案中存在 Bentley 2 汽車,因此它會列印已找到。

Bentley 2 car is found

按屬性查詢元素

Element 介面的getAttributeByName()方法接受 QName 物件(即屬性的限定 XML 名稱)並返回 Attribute 物件。此外,Attribute 介面的getValue()方法用於以字串的形式獲取屬性的值。

示例 1

我們在前一個示例中使用的 cars.xml 檔案在以下QueryAttributes.java程式中被解析,以計算 XML 檔案中存在的 Bentley 汽車數量。每次 carname 元素的 company 屬性等於 Bentley 時,計數器變數都會遞增。

import java.io.FileReader;
import javax.xml.namespace.QName;
import javax.xml.stream.XMLEventReader;
import javax.xml.stream.XMLInputFactory;
import javax.xml.stream.XMLStreamConstants;
import javax.xml.stream.events.Attribute;
import javax.xml.stream.events.StartElement;
import javax.xml.stream.events.XMLEvent;

public class QueryExample2 {
   public static void main(String args[]) {
      try {
    	  
         //Creating XMLInputFactory instance
      	 XMLInputFactory factory = XMLInputFactory.newInstance();
      	 
      	 //Reading the XML
      	 FileReader fileReader = new FileReader("cars.xml");
      	 
      	 //Parsing the XML
         XMLEventReader eventReader =
           factory.createXMLEventReader(fileReader);
         
          //Querying the XML document
          int count=0;
          while(eventReader.hasNext()) {
        	  XMLEvent event = eventReader.nextEvent();
        	  if(event.getEventType()==XMLStreamConstants.START_ELEMENT) {
              	StartElement element=event.asStartElement();
              	QName qname=new QName("company");
                Attribute attr=element.getAttributeByName(qname);
                if(attr!=null && attr.getValue().equals("Bentley"))
                   count++;
              }
          }
          System.out.println("No.of Bentley cars found : " + count);          
	  } catch(Exception e) {
		   e.printStackTrace();
      }
   }
}

顯示 XML 檔案中 Bentley 汽車的數量。

No.of Bentley cars found : 3

示例 2

在這個例子中,我們將根據學生的學號檢索特定學生的資訊。這是我們需要查詢的student.xml檔案:

<?xml version = "1.0"?>
<class>
   <student rollno = "393">
      <firstname>dinkar</firstname>
      <lastname>kad</lastname>
      <nickname>dinkar</nickname>
      <marks>85</marks>
   </student>
   
   <student rollno = "493">
      <firstname>Vaneet</firstname>
      <lastname>Gupta</lastname>
      <nickname>vinni</nickname>
      <marks>95</marks>
   </student>
   
   <student rollno = "593">
      <firstname>jasvir</firstname>
      <lastname>singn</lastname>
      <nickname>jazz</nickname>
      <marks>90</marks>
   </student>
</class>

在下面的QueryStudent.java程式中,我們檢查了 START_ELEMENT、CHARACTERS 和 END_ELEMENT 事件,並相應地執行了操作。當學號屬性等於 393 時,我們將列印學生的全部資訊。

import java.io.FileReader;
import java.util.Iterator;
import javax.xml.stream.XMLEventReader;
import javax.xml.stream.XMLInputFactory;
import javax.xml.stream.XMLStreamConstants;
import javax.xml.stream.events.Attribute;
import javax.xml.stream.events.Characters;
import javax.xml.stream.events.EndElement;
import javax.xml.stream.events.StartElement;
import javax.xml.stream.events.XMLEvent;

public class QueryStudent {
   public static void main(String[] args) {
      boolean bFirstName = false;
      boolean bLastName = false;
      boolean bNickName = false;
      boolean bMarks = false;
      boolean isRequestRollNo = false;
      
      try {
    	  
    	 //Creating XMLInputFactory instance
       	 XMLInputFactory factory = XMLInputFactory.newInstance();
       	 
       	 //Reading the XML
       	 FileReader fileReader = new FileReader("student.xml");
       	 
       	 //Parsing the XML
         XMLEventReader eventReader =
            factory.createXMLEventReader(fileReader);
         
    	 //Querying the XML 
         String requestedRollNo = "393";
         while(eventReader.hasNext()) {
            XMLEvent event = eventReader.nextEvent();
            
            switch(event.getEventType()) {
               case XMLStreamConstants.START_ELEMENT:
                  StartElement startElement = event.asStartElement();
                  String qName = startElement.getName().getLocalPart();
                  
               if (qName.equalsIgnoreCase("student")) {
                  Iterator<Attribute> attributes = startElement.getAttributes();
                  String rollNo = attributes.next().getValue();
                     
                  if(rollNo.equalsIgnoreCase(requestedRollNo)) {
                     System.out.println("Start Element : student");
                     System.out.println("Roll No : " + rollNo);
                     isRequestRollNo = true;
                  }
               } else if (qName.equalsIgnoreCase("firstname")) {
                  bFirstName = true;
               } else if (qName.equalsIgnoreCase("lastname")) {
                  bLastName = true;
               } else if (qName.equalsIgnoreCase("nickname")) {
                  bNickName = true;
               }
               else if (qName.equalsIgnoreCase("marks")) {
                  bMarks = true;
               }
               break;
               
               case XMLStreamConstants.CHARACTERS:
                  Characters characters = event.asCharacters();
               
               if(bFirstName && isRequestRollNo) {
                  System.out.println("First Name: " + characters.getData());
                  bFirstName = false;
               }
               if(bLastName && isRequestRollNo) {
                  System.out.println("Last Name: " + characters.getData());
                  bLastName = false;
               }
               if(bNickName && isRequestRollNo) {
                  System.out.println("Nick Name: " + characters.getData());
                  bNickName = false;
               }
               if(bMarks && isRequestRollNo) {
                  System.out.println("Marks: " + characters.getData());
                  bMarks = false;
               }
               break;
               
               case XMLStreamConstants.END_ELEMENT:
                  EndElement endElement = event.asEndElement();
                  
               if(endElement.getName().getLocalPart().equalsIgnoreCase(
                  "student") && isRequestRollNo) {
                  System.out.println("End Element : student");
                  System.out.println();
                  isRequestRollNo = false;
               }
               break;
            }
         }
      } catch (Exception e) {
         e.printStackTrace();
      }
   }
}

顯示學號為 393 的學生的全部資訊。

Start Element : student
Roll No : 393
First Name: dinkar
Last Name: kad
Nick Name: dinkar
Marks: 85
End Element : student
廣告