Использование JXPath для запроса списка
У меня есть простой класс (для тестирования), который я пытаюсь сделать запрос против использования JXPath.
Я создаю список различных животных объектов и хочу получить итератор для:
Все животные, где тип ='CAT'
Все животные, где 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"
Надеюсь, это немного поможет.
Я не уверен, что не работает в вышеупомянутом. Однако пара моментов.
- Я не уверен, что перечисления работают так, как вы ожидаете. Ваш запрос JXPath запрашивает тип, где
getType()
вернет перечисление, а не строку. Я не уверен, что JXPath такой умный. Перечисления. [type='CAT']
само по себе не является выражением XPath. Я бы ожидал что-то вроде//*[type='CAT']
Поэтому я бы попробовал следующее:
- реализовать метод getTypeName(), который возвращает имя перечисления, например, CAT/DOG и т. д., и запрашивать его
- Попробуйте XPath из
//*
(без каких-либо предикатов), чтобы подтвердить, что JXpath будет запрашивать ArrayList, как и ожидалось.