Я думаю, что Digester не преуспел в разборе устаревшего xml. Но, к счастью, он достаточно расширяемый, чтобы его можно было настроить для работы в некоторых из этих случаев.
Мне удалось решить эту проблему только с помощью API Java, а не с правилами XML. Важно использовать класс ExtendedBaseRules в Digester, чтобы соответствовать шаблонам row / row / * и создать подкласс для стандартного CallParamRule, чтобы передать имя тега в качестве первого аргумента методу addAddressLine () класса Address (я переименовал метод set () в вашем классе Address для addAddressLine ()).
Digester digester = new Digester();
digester.setRules(new ExtendedBaseRules());
digester.setValidating( false );
digester.addObjectCreate("rows", Addresses.class);
digester.addObjectCreate( "rows/row", Address.class );
digester.addSetNext( "rows/row", "add");
digester.addSetProperties("rows/row");
digester.addCallMethod("rows/row/*", "addAddressLine", 2);
digester.addRule("rows/row/*", new TagNameAwarePathCallParamRule(0));
digester.addCallParam("rows/row/*", 1);
Addresses addresses = (Addresses) digester.parse(new File(FILE_TO_PARSE));
Класс TagNameAwarePathCallParamRule, который я реализовал для этой цели:
public class TagNameAwarePathCallParamRule extends CallParamRule {
public TagNameAwarePathCallParamRule(int paramIndex) {
super(paramIndex);
}
public void end(String namespace, String name) {
if (bodyTextStack != null && !bodyTextStack.empty()) {
// what we do now is push one parameter onto the top set of
// parameters
Object parameters[] = (Object[]) digester.peekParams();
parameters[paramIndex] = name;
}
}
}
Классы, которые я использовал для хранения результатов анализа:
public class Addresses {
private List<Address> addresses = new ArrayList<Address>();
public void add(Address a) {
addresses.add(a);
}
public List<Address> getAddresses() {
return Collections.unmodifiableList(addresses);
}
public String toString() {
StringBuilder sb = new StringBuilder();
for (Address a : addresses) {
sb.append("Address: ").append(a.toString()).append(',');
}
return sb.toString();
}
}
public class Address {
Map<String,String> c= new HashMap<String,String>();
public void addAddressLine(String name, String value){
c.put(name, value);
}
public String toString(){
return c.toString();
}
}