Предикаты в Hibernate-Spatial + PostGIS для поиска в радиусе миль с широтой / долготой

Я пытаюсь использовать предикаты в Hibernate-Spatial для поиска в радиусе X миль (определяется пользователем).

Хотя такого рода вопросы задавались раньше, только один раз с предикатами и, возможно, не актуальны, поскольку не работают. И я должен уметь использовать предикаты, а НЕ HQL.

С помощью другого ответа я нашел, как настроить предикат для ключевого слова WITHIN в PostGIS, но мне не удалось запустить запрос.

После настройки в конце я указываю вывод ошибок и версии пакета.

Вот что у меня в схеме:

CREATE EXTENSION Postgis;

CREATE TABLE "user" (
    id               bigint  NOT NULL,
    search_radius    int,
    latitude         double precision,
    longitude        double precision,
    location         geography(point, 4326),

    PRIMARY KEY (id)
);

В моей модели:

import com.vividsolutions.jts.geom.Point;
import com.vividsolutions.jts.io.ParseException;
import com.vividsolutions.jts.io.WKTReader;
import org.hibernate.annotations.Type;

@Entity
@Table(name = "`user`")
public class User implements UserDetails {
    @Id
    private Long id;
    private Integer searchRadius = 125;
    private Double latitude;
    private Double longitude;
    @Column(name = "location", columnDefinition = "geography(Point, 4326)", nullable = true)
    @Type(type = "jts_geometry")
    private Point location;

    ...

    private void updateLocation() {
        if (latitude != null && longitude != null) {
            try {
                location = (Point) new WKTReader().read("POINT(" + latitude + " " + longitude + ")");
                location.setSRID(Location.SRID);
            } catch (ParseException e) {
                e.printStackTrace();
                throw new RuntimeException(e);
            }
        }
    }
}

Где местоположение обновления вызывается внутри установщиков широты / долготы.

Предикат, который я получил из Используя JPA Criteria Api и спящий пространственный 4 вместе

import com.vividsolutions.jts.geom.Geometry;
import com.vividsolutions.jts.geom.Point;
import java.io.Serializable;
import javax.persistence.criteria.Expression;
import org.hibernate.jpa.criteria.CriteriaBuilderImpl;
import org.hibernate.jpa.criteria.ParameterRegistry;
import org.hibernate.jpa.criteria.Renderable;
import org.hibernate.jpa.criteria.compile.RenderingContext;
import org.hibernate.jpa.criteria.predicate.AbstractSimplePredicate;

public class WithinPredicate extends AbstractSimplePredicate implements Expression<Boolean>, Serializable {
    private final Expression<Point> matchExpression;
    private final Expression<Geometry> area;

    public WithinPredicate(CriteriaBuilderImpl criteriaBuilder, Expression<Point> matchExpression, Expression<Geometry> area) {
        super(criteriaBuilder);
        this.matchExpression = matchExpression;
        this.area = area;
    }

    public Expression<Point> getMatchExpression() {
        return matchExpression;
    }

    public Expression<Geometry> getArea() {
        return area;
    }

    public void registerParameters(ParameterRegistry registry) {
        // Nothing to register
    }

    @Override
    public String render(boolean isNegated, RenderingContext renderingContext) {
        StringBuilder buffer = new StringBuilder();
        buffer.append(" within(")
            .append(((Renderable) getMatchExpression()).render(renderingContext))
            .append(", ")
            .append(((Renderable) getArea()).render(renderingContext))
            .append(" ) = true ");
        String bo = buffer.toString();

        return bo;
    }
}

И создаем запрос, используя WithinPredicate и фабрику окружностей.

import com.xxxx.repository.WithinPredicate;
import com.vividsolutions.jts.geom.Coordinate;
import com.vividsolutions.jts.geom.Geometry;
import com.vividsolutions.jts.util.GeometricShapeFactory;

@Service
class ... {
    public List<User> fetchFor(User user) {
        ...
        Geometry radius = createCircle(user.getLatitude(), user.getLongitude(), user.getSearchRadius());
        radius.setSRID(Location.SRID);
        p = b.and(p, new WithinPredicate(
            (CriteriaBuilderImpl) b,
            u.get(User_.location),
            new LiteralExpression<Geometry>((CriteriaBuilderImpl) b, radius)
        ));
    }

    private static Geometry createCircle(double x, double y, final double RADIUS) {
        GeometricShapeFactory shapeFactory = new GeometricShapeFactory();
        shapeFactory.setNumPoints(32);
        shapeFactory.setCentre(new Coordinate(x, y));//there are your coordinates
        shapeFactory.setSize(RADIUS * 2);//this is how you set the radius
        return shapeFactory.createCircle().getBoundary();
    }
}

К сожалению, на выходе я получаю следующую ошибку:

2016-09-09 14:03:30.862  WARN 56393 --- [           main] o.h.engine.jdbc.spi.SqlExceptionHelper   : SQL Error: 0, SQLState: 42883
2016-09-09 14:03:30.862 ERROR 56393 --- [           main] o.h.engine.jdbc.spi.SqlExceptionHelper   : ERROR: function st_within(geography, bytea) does not exist
  Hint: No function matches the given name and argument types. You might need to add explicit type casts.
  Position: 336

Спящая пространственная версия в градиенте:

compile('org.hibernate:hibernate-spatial:5.0.9.Final')

Я подозреваю, что проблема может быть связана с тем, что модель / схема настроены на тип «География (точка, 4326)», но shapeFactory создает радиус типа «Геометрия (точка, 4326)».

Очень признателен за любую помощь в этом. Спасибо.


person Reece Fowell    schedule 12.09.2016    source источник


Ответы (1)


Я прочитал только часть вашего вопроса, однако некоторые вещи вызвали несколько красных флажков:

  • Вам нужно перевернуть порядок оси на (X Y) или (долгота, широта)
  • ST_Within существует только для geometry типов, но не для geography
  • Чтобы найти точки в пределах региона, не создавайте несовершенные точки с буферизацией.
  • Вы найдете ST_DWithin более быстрым и хорошо поддерживаемым для типа geography; для радиуса просто сократите свои мили в метры
person Mike T    schedule 14.09.2016