Вы можете изменить обработчик содержимого во время обработки, что позволяет вам поддерживать состояние и выполнять контекстную обработку.Например, рассмотрим этот автономный пример:
import java.util.ArrayList;
import java.util.List;
import javax.xml.parsers.SAXParser;
import javax.xml.parsers.SAXParserFactory;
import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
import org.xml.sax.XMLReader;
import org.xml.sax.helpers.DefaultHandler;
public class SAXHandler extends DefaultHandler {
private XMLReader reader;
private List<Person> people = new ArrayList<Person>();
public static void main(String[] args) {
try {
SAXParserFactory factory = SAXParserFactory.newInstance();
SAXParser saxParser = factory.newSAXParser();
SAXHandler tester = new SAXHandler(saxParser.getXMLReader());
saxParser.parse("workbook.xml", tester);
System.out.println(tester.people);
} catch (Exception e) {
e.printStackTrace();
}
}
public SAXHandler(XMLReader reader) {
this.reader = reader;
}
public Person addPerson(Person person) {
this.people.add(person);
return person;
}
public void startElement(String uri, String localName, String qName,
Attributes attributes) throws SAXException {
if ("person".equals(qName)) {
String name = attributes.getValue("name");
this.reader.setContentHandler(new PersonHandler(this.reader, this,
name));
}
System.out.println("Start Element: " + qName);
}
}
Обратите внимание, что мы setContentHandler
всякий раз, когда встречаем элемент person
.PersonHandler
выглядит следующим образом:
import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
import org.xml.sax.XMLReader;
import org.xml.sax.helpers.DefaultHandler;
public class PersonHandler extends DefaultHandler {
private Person currPerson;
private StringBuilder content;
private XMLReader reader;
private SAXHandler parentHandler;
public PersonHandler(XMLReader reader, SAXHandler parentHandler, String name) {
this.reader = reader;
this.parentHandler = parentHandler;
this.currPerson = new Person(name);
this.content = new StringBuilder();
}
public void startElement(String uri, String localName, String qName,
Attributes attributes) throws SAXException {
this.content.setLength(0);
System.out.println("PersonHandler::Start Element: " + qName);
}
public void endElement(String uri, String localName, String qName)
throws SAXException {
if ("age".equals(qName)) {
this.currPerson.setAge(this.content.toString());
} else if ("city".equals(qName)) {
this.currPerson.setCity(this.content.toString());
} else if ("person".equals(qName)) {
this.parentHandler.addPerson(this.currPerson);
this.reader.setContentHandler(this.parentHandler);
}
System.out.println("PersonHandler::End Element: " + qName);
}
public void characters(char ch[], int start, int length)
throws SAXException {
this.content.append(ch, start, length);
}
}
Обратите внимание, что мы переустанавливаем обработчик содержимого на родительский обработчик всякий раз, когда мы завершаем обработку person
.
Для полноты вот (очень) минимальный Person
:
public class Person {
private String name;
private String age;
private String city;
public Person(String name) {
this.name = name;
}
public void setAge(String string) {
this.age = string;
}
public void setCity(String string) {
this.city = string;
}
public String toString() {
return this.name + " " + this.age + " " + this.city;
}
}
Этот код выведет:
Start Element: root
Start Element: person
PersonHandler::Start Element: age
PersonHandler::End Element: age
PersonHandler::Start Element: city
PersonHandler::End Element: city
PersonHandler::End Element: person
Start Element: person
PersonHandler::Start Element: age
PersonHandler::End Element: age
PersonHandler::Start Element: city
PersonHandler::End Element: city
PersonHandler::End Element: person