Вот реализация на Java для создания запроса на основе местоположения на устройстве Android. Идея исходит от KennyTM (см. Принятый ответ) и подразумевает добавление 4 столбцов в таблице для хранения значений синуса и косинуса широты и долготы.
Вот код подготовки данных для таблицы «Магазин» во время вставки:
public static void injectLocationValues(ContentValues values, double latitude, double longitude) {
values.put(LocationColumns.LATITUDE, latitude);
values.put(LocationColumns.LONGITUDE, longitude);
values.put(LocationColumns.COSLAT, Math.cos(MathUtil.deg2rad(latitude)));
values.put(LocationColumns.SINLAT, Math.sin(MathUtil.deg2rad(latitude)));
values.put(LocationColumns.COSLNG, Math.cos(MathUtil.deg2rad(longitude)));
values.put(LocationColumns.SINLNG, Math.sin(MathUtil.deg2rad(longitude)));
}
public static double deg2rad(double deg) {
return (deg * Math.PI/180.0);
}
Вы можете создать свой прогноз с помощью следующей функции:
/**
* Build query based on distance using spherical law of cosinus
*
* d = acos(sin(lat1).sin(lat2)+cos(lat1).cos(lat2).cos(long2−long1)).R
* where R=6371 and latitudes and longitudes expressed in radians
*
* In Sqlite we do not have access to acos() sin() and lat() functions.
* Knowing that cos(A-B) = cos(A).cos(B) + sin(A).sin(B)
* We can determine a distance stub as:
* d = sin(lat1).sin(lat2)+cos(lat1).cos(lat2).(cos(long2).cos(long1)+sin(long2).sin(long1))
*
* First comparison point being fixed, sin(lat1) cos(lat1) sin(long1) and cos(long1)
* can be replaced by constants.
*
* Location aware table must therefore have the following columns to build the equation:
* sinlat => sin(radians(lat))
* coslat => cos(radians(lat))
* coslng => cos(radians(lng))
* sinlng => sin(radians(lng))
*
* Function will return a real between -1 and 1 which can be used to order the query.
* Distance in km is after expressed from R.acos(result)
*
* @param latitude, latitude of search
* @param longitude, longitude of search
* @return selection query to compute the distance
*/
public static String buildDistanceQuery(double latitude, double longitude) {
final double coslat = Math.cos(MathUtil.deg2rad(latitude));
final double sinlat = Math.sin(MathUtil.deg2rad(latitude));
final double coslng = Math.cos(MathUtil.deg2rad(longitude));
final double sinlng = Math.sin(MathUtil.deg2rad(longitude));
//@formatter:off
return "(" + coslat + "*" + LocationColumns.COSLAT
+ "*(" + LocationColumns.COSLNG + "*" + coslng
+ "+" + LocationColumns.SINLNG + "*" + sinlng
+ ")+" + sinlat + "*" + LocationColumns.SINLAT
+ ")";
//@formatter:on
}
Это впрыснуть ответную колонку с расстоянием, на котором вам нужно применить следующую формулу для преобразования в километрах:
public static double convertPartialDistanceToKm(double result) {
return Math.acos(result) * 6371;
}
Если вы хотите заказать свой запрос, используя частичное расстояние, вам необходимо заказать DESC, а не ASC.
У вас есть два варианта расчета, за пределами базы данных или проектируйте координаты на эллипсоиде, которые доставят вам счетчики и позволят использовать формулу правильного расстояния в качестве приближения. –
Я открыт для этого решения, можете ли вы подробно рассказать об этом в ответ. – Pentium10
Для начинающих, таких как я, вся формула называется «Сферический закон косинусов» и описывается здесь: http://www.movable-type.co.uk/scripts/latlong.html –