MongoRepository findByCreatedAtBetween не возвращает точные результаты

Моя структура документа в Монго такова:

db.user.find ()

{
        "_id" : ObjectId("560fa46930a8e74be720009a"),
        "createdAt" : ISODate("2015-10-03T09:47:56.333Z"),
        "message" : "welcome",
}
{
            "_id" : ObjectId("560fa46930a8e723e720009a"),
            "createdAt" : ISODate("2015-10-03T09:48:25.048Z"),
            "message" : "thank you"
}

Когда я использую приведенный ниже запрос в моей оболочке Mongo для поиска документов между двумя заданными временными метками, я получаю правильный результат:

db.user.find({createdAt:{$gte:ISODate("2015-10-03T09:40:25.048Z"),$lte:ISODate("2015-10-03T09:50:56.333Z")}})

Я использую MongoRepository с Spring в моем REST-сервисе для работы с драйвером Java. Ниже находится хранилище пользователя:

public interface UserRepository extends MongoRepository<User, String> 
{
       ArrayList<User> findbyCreatedAtBetween(Date d1, Date d2);

}

Когда я звоню, сделайте следующий звонок в моем сервисе, он не вернет никакого результата.userRepository.findbyCreatedAtBetween(2015-10-03T09:40:25.048Z, 2015-10-03T09:50:29.006Z)

Однако он возвращает результат, когда я задаю d1 как предыдущий день: userRepository.findbyCreatedAtBetween(2015-10-02T09:40:25.048Z, 2015-10-03T09:50:29.006Z)

Любые идеи о том, как я могу решить эту проблему? Пожалуйста помоги!

1 ответ

Решение

Разбивая его, запрос с ключевым словом Between выполняется с базой данных MongoDB с логическим результатом {"createdAt" : {"$gt" : d1, "$lt" : d2}} так что есть вероятность, что вы не получаете документы, которые имеют createdAt дата включительно в пределах данного диапазона дат, т.е. d1 < createdAt < d2 поскольку данный диапазон дат не удовлетворяет критериям. Для справки вот некоторые из интерпретаций методов запроса:

Поддерживаемые ключевые слова для методов запроса

Keyword     Sample                              Logical result
After       findByBirthdateAfter(Date date)     {"birthdate" : {"$gt" : date}}
Before      findByBirthdateBefore(Date date)    {"birthdate" : {"$lt" : date}}
Between     findByAgeBetween(int from, int to)  {"age" : {"$gt" : from, "$lt" : to}}

В качестве обходного пути вы можете использовать @Query аннотаций. Я не проверял это, но вы можете попробовать следующий пример реализации пользовательского запроса:

public interface UserRepository extends MongoRepository<User, String>  {
    @Query(value = "{ 'createdAt' : {$gte : ?0, $lte: ?1 }}")
    public ArrayList<User> findbyCreatedAtBetween(Date from, Date to);
}

Если вышеупомянутое не работает для вас, создайте пользовательский интерфейс и ваш класс реализации для выполнения пользовательского запроса. Например, создайте интерфейс с именем, которое добавляет Custom:

public interface UserRepositoryCustom {
    public List<User> findbyCreatedAtBetween(Date from, Date to); 
}

Изменить UserRepository и добавить UserRepositoryCustom расширяемый интерфейс:

@Repository
public interface UserRepository extends UserRepositoryCustom, MongoRepository {

}

Создайте свой класс реализации для реализации методов, определенных в UserRepositoryCustom интерфейс.

public class UserRepositoryImpl implements UserRepositoryCustom {

    @Autowired
    MongoTemplate mongoTemplate;

    @Override
    public ArrayList<User> findbyCreatedAtBetween(Date from, Date to) {
        return mongoTemplate.find(
            Query.addCriteria(Criteria.where("createdAt").gte(from).lte(to));
    }
}
Другие вопросы по тегам