Использование JXPath для запроса списка

У меня есть простой класс (для тестирования), который я пытаюсь сделать запрос против использования 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 ответа

Решение

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

/*[type='CAT']   

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

Я думаю, что вы после чего-то больше

/*/*[type='CAT']

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

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

//*[type='CAT'] 

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

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

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

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

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

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