Подстановочный знак в XML-правилах Digester - PullRequest
1 голос
/ 29 сентября 2010

Я видел разные темы, обсуждающие проблему, которая у меня есть. Вот несколько примеров: http://www.mailinglistarchive.com/commons-user@jakarta.apache.org/msg05061.html, Дайджест: извлечение имени узла ).

Тем не менее, я до сих пор не могу найти решение этой проблемы. Вот мои данные XML:

<rows>
   <row>
        <firstname>Paul</firstname>
        <lastname>Moris</lastname>
   </row>
   <row>
        <firstname>John</firstname>
        <lastname>Aiyer</lastname>
        <age>35</age>
   </row>
</rows>

То, что я хочу, это определить правила XML, которые позволили бы мне отображать каждую строку в карту. Я не могу сопоставить элементы, ссылаясь на их имена, потому что не все возможные элементы известны заранее.

Я надеялся, что что-то подобное позволит мне сделать это:

<digester-rules>
     <pattern value="rows/row">
         <object-create-rule classname="Address"/>
         <set-next-rule methodname="add" paramtype="java.lang.Object"/>
         <set-properties-rule/>

         <pattern value="*">
             <call-method-rule methodname="set" paramcount="2"/>  
             <call-param-rule paramnumber='0'/>
             <call-param-rule paramnumber='1'/>
         </pattern>
     </pattern>
</digester-rules>

Реализация адреса:

public class Address {
   Map<String,String> c= new HashMap<String,String>();

   public void set(String name, String value){
   c.put(name, value);
   }

  public String toString(){
   return c.toString();
  }
}

К сожалению, когда я запускаю этот код, я получаю два адреса, которые созданы, но с пустой основной картой. Когда я использую ExtendedBaseRules, ничто даже не совпадает.

Любая помощь будет высоко ценится.

Макс.

1 Ответ

1 голос
/ 04 ноября 2010

Я думаю, что 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();
  }
}
...