Получить данные
Существует много способов получения собранных данных о местоположении. Здесь мы описываем два метода получения данных для использования с функцией привязки к дорогам Roads API .
ГПХ
GPX — это открытый формат на основе XML для обмена маршрутами, треками и точками маршрута, полученными с помощью GPS-устройств. В этом примере используется парсер XmlPull , легкий парсер XML, доступный как для серверов Java, так и для мобильных сред.
/** * Parses the waypoint (wpt tags) data into native objects from a GPX stream. */ private List<LatLng> loadGpxData(XmlPullParser parser, InputStream gpxIn) throws XmlPullParserException, IOException { // We use a List<> as we need subList for paging later List<LatLng> latLngs = new ArrayList<>(); parser.setInput(gpxIn, null); parser.nextTag(); while (parser.next() != XmlPullParser.END_DOCUMENT) { if (parser.getEventType() != XmlPullParser.START_TAG) { continue; } if (parser.getName().equals("wpt")) { // Save the discovered latitude/longitude attributes in each <wpt>. latLngs.add(new LatLng( Double.valueOf(parser.getAttributeValue(null, "lat")), Double.valueOf(parser.getAttributeValue(null, "lon")))); } // Otherwise, skip irrelevant data } return latLngs; }
Вот некоторые необработанные данные GPX, загруженные на карту.
Службы определения местоположения Android
Лучший способ сбора данных GPS с устройства Android зависит от вашего варианта использования. Ознакомьтесь с обучающим классом Android по получению обновлений местоположения , а также с примерами местоположения Google Play на GitHub .
Обрабатывать длинные пути
Поскольку функция привязки к дорогам определяет местоположение на основе полного пути, а не отдельных точек, необходимо проявлять осторожность при обработке длинных путей (то есть путей, превышающих лимит в 100 точек на запрос).
Чтобы рассматривать отдельные запросы как один длинный путь, следует включить некоторое перекрытие, так чтобы конечные точки из предыдущего запроса были включены как первые точки последующего запроса. Количество включаемых точек зависит от точности ваших данных. Для запросов с низкой точностью следует включить больше точек.
В этом примере используется Java-клиент для служб Google Maps для отправки постраничных запросов, а затем данные, включая интерполированные точки, объединяются в возвращаемый список.
/** * Snaps the points to their most likely position on roads using the Roads API. */ private List<SnappedPoint> snapToRoads(GeoApiContext context) throws Exception { List<SnappedPoint> snappedPoints = new ArrayList<>(); int offset = 0; while (offset < mCapturedLocations.size()) { // Calculate which points to include in this request. We can't exceed the API's // maximum and we want to ensure some overlap so the API can infer a good location for // the first few points in each request. if (offset > 0) { offset -= PAGINATION_OVERLAP; // Rewind to include some previous points. } int lowerBound = offset; int upperBound = Math.min(offset + PAGE_SIZE_LIMIT, mCapturedLocations.size()); // Get the data we need for this page. LatLng[] page = mCapturedLocations .subList(lowerBound, upperBound) .toArray(new LatLng[upperBound - lowerBound]); // Perform the request. Because we have interpolate=true, we will get extra data points // between our originally requested path. To ensure we can concatenate these points, we // only start adding once we've hit the first new point (that is, skip the overlap). SnappedPoint[] points = RoadsApi.snapToRoads(context, true, page).await(); boolean passedOverlap = false; for (SnappedPoint point : points) { if (offset == 0 || point.originalIndex >= PAGINATION_OVERLAP - 1) { passedOverlap = true; } if (passedOverlap) { snappedPoints.add(point); } } offset = upperBound; } return snappedPoints; }
Вот данные сверху после запуска запросов snap to roads. Красная линия — это необработанные данные, а синяя линия — это привязанные данные.
Эффективное использование квоты
Ответ на запрос привязки к дорогам включает список идентификаторов мест, которые сопоставляются с указанными вами точками, возможно, с дополнительными точками, если вы установите interpolate=true
.
Чтобы эффективно использовать вашу разрешенную квоту для запроса ограничений скорости, вам следует запрашивать только уникальные идентификаторы мест в вашем запросе. В этом примере используется Java-клиент для Google Maps Services для запроса ограничений скорости из списка идентификаторов мест.
/** * Retrieves speed limits for the previously-snapped points. This method is efficient in terms * of quota usage as it will only query for unique places. * * Note: Speed limit data is only available for requests using an API key enabled for a * Google Maps APIs Premium Plan license. */ private Map<String, SpeedLimit> getSpeedLimits(GeoApiContext context, List<SnappedPoint> points) throws Exception { Map<String, SpeedLimit> placeSpeeds = new HashMap<>(); // Pro tip: Save on quota by filtering to unique place IDs. for (SnappedPoint point : points) { placeSpeeds.put(point.placeId, null); } String[] uniquePlaceIds = placeSpeeds.keySet().toArray(new String[placeSpeeds.keySet().size()]); // Loop through the places, one page (API request) at a time. for (int i = 0; i < uniquePlaceIds.length; i += PAGE_SIZE_LIMIT) { String[] page = Arrays.copyOfRange(uniquePlaceIds, i, Math.min(i + PAGE_SIZE_LIMIT, uniquePlaceIds.length)); // Execute! SpeedLimit[] placeLimits = RoadsApi.speedLimits(context, page).await(); for (SpeedLimit sl : placeLimits) { placeSpeeds.put(sl.placeId, sl); } } return placeSpeeds; }
Ниже приведены данные, содержащие ограничения скорости, отмеченные для каждого уникального идентификатора места.
Взаимодействие с другими API
Одним из преимуществ возврата идентификаторов мест в ответах snap to roads является то, что вы можете использовать идентификатор места во многих API платформы Google Maps. В этом примере используется Java-клиент для служб Google Maps для геокодирования места, возвращаемого из приведенного выше запроса snap to road.
/** * Geocodes a snapped point using the place ID. */ private GeocodingResult geocodeSnappedPoint(GeoApiContext context, SnappedPoint point) throws Exception { GeocodingResult[] results = GeocodingApi.newRequest(context) .place(point.placeId) .await(); if (results.length > 0) { return results[0]; } return null; }
Здесь маркер ограничения скорости аннотирован адресом из API геокодирования.
Пример кода
Соображения
Код, поддерживающий этот документ, доступен в виде отдельного приложения Android для иллюстративных целей. На практике вам не следует распространять ключи API на стороне сервера в приложении Android, поскольку ваш ключ не может быть защищен от несанкционированного доступа третьей стороны. Вместо этого, чтобы защитить ваши ключи, вам следует развернуть код API в качестве прокси-сервера на стороне сервера и заставить ваше приложение Android отправлять запросы с помощью прокси-сервера, чтобы убедиться, что запросы авторизованы.
Скачать
Загрузите код с GitHub .