Использование JXPath для запроса списка - PullRequest
2 голосов
/ 22 мая 2009

У меня есть простой класс (для целей тестирования), который я пытаюсь выполнить с помощью JXPath.

Я создаю список различных объектов животных и хочу получить итератор для:

  1. Все животные, где тип = 'CAT'

  2. Все животные, где numLegs = 4

Вот простой класс:

public class Animal {

    private UUID uuid;
    private int numLegs;
    private AnimalType type;

    public enum AnimalType {
        CHICKEN, FROG, DOG, CAT
    }

    public Animal(AnimalType type) {
        this.type = type;
        uuid = UUID.randomUUID();
        switch (type) {
        case CHICKEN:
            numLegs = 2;
            break;
        case FROG:
            numLegs = 2;
            break;
        case DOG:
            numLegs = 4;
            break;
        case CAT:
            numLegs = 4;
            break;
        }
    }

    public UUID getUuid() {
        return uuid;
    }

    public int getNumLegs() {
        return numLegs;
    }

    public AnimalType getType() {
        return type;
    }

    public String toString(){
        return "UUID: "+uuid+", Animal: "+type+ ", Legs: "+numLegs;
    }

}

Вот метод, который я использую для составления списка животных для запроса:

private static List<Animal> getAnimals(int numAnimals) {
    ArrayList<Animal> animals = new ArrayList<Animal>();
    for(int i = 0; i<numAnimals; i++){
        if(i%4==0){
            animals.add(new Animal(AnimalType.CAT));
        }
        else if(i%3==0){
            animals.add(new Animal(AnimalType.DOG));
        }
        else if(i%2==0){
            animals.add(new Animal(AnimalType.FROG));
        }
        else{
            animals.add(new Animal(AnimalType.CHICKEN));
        }

    }

    return animals;
}

Вот как я пытаюсь выполнить запрос:

public static void main(String[] args){
    List<Animal> animals = getAnimals(10000);

    JXPathContext animsContext = JXPathContext.newContext(animals);

    Iterator<BeanPointer> iter = 
        animsContext.iteratePointers("/*[type='CAT']");

    List<Animal> cats = new ArrayList<Animal>();


    while(iter.hasNext()){
        cats.add((Animal) iter.next().getParent().getValue());
    }
    System.out.println("Number of Cats is: "+cats.size());
}

Эта часть:

    Iterator<BeanPointer> iter = 
        animsContext.iteratePointers("/*[type='CAT']");

не работает. Что я делаю неправильно? Я не могу заставить его работать на / * [numLegs = 4].

Ответы [ 2 ]

5 голосов
/ 29 мая 2009

Я никогда не использовал JXPath, но я ожидал, что будет корневой узел, в этом случае

/*[type='CAT']   

будет означать «дать мне корневой узел, только если он имеет атрибут типа, равный CAT»

Я думаю, что вы после чего-то похожи на

/*/*[type='CAT']

, который читается как "дать мне узлы прямо под корневым узлом, которые имеют тип атрибута, равный CAT"

предложение Брайана

//*[type='CAT'] 

читается как «дать мне все узлы в дереве, которые имеют тип атрибута, равный CAT»

Надеюсь, это немного поможет.

2 голосов
/ 22 мая 2009

Я не уверен, что не работает в выше. Однако пара моментов.

  1. Я не уверен, что перечисления работают так, как вы ожидаете. Ваш запрос JXPath запрашивает тип, в котором getType() будет возвращать перечисление, а не строку. Я не уверен, что JXPath такой умный. Перечисления.
  2. [type='CAT'] само по себе не является выражением XPath. Я бы ожидал что-то вроде //*[type='CAT']

Поэтому я бы попробовал следующее:

  1. реализовать метод getTypeName (), который возвращает имя перечисления, например, CAT / DOG и т. Д. И запрос с использованием этого
  2. Попробуйте XPath //* (без предикатов), чтобы подтвердить, что JXpath будет запрашивать ArrayList, как и ожидалось.
...