Acquisire dati
Esistono molti modi per ottenere i dati sulla posizione raccolti. Di seguito descriviamo due tecniche per acquisire i dati da utilizzare con la funzionalità di allineamento alle strade dell' Roads API.
GPX
GPX è un formato aperto basato su XML per la condivisione di route, tracce e waypoint acquisiti dai dispositivi GPS. Questo esempio utilizza il XmlPull, un parser XML leggero disponibile sia per gli ambienti server Java che per quelli mobile.
/** * 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; }
Di seguito sono riportati alcuni dati GPX non elaborati caricati su una mappa.

Servizi di localizzazione Android
Il modo migliore per acquisire i dati GPS da un dispositivo Android varia a seconda del caso d'uso. Dai un'occhiata alla classe di formazione Android su come ricevere aggiornamenti sulla posizione, nonché agli esempi di localizzazione di Google Play su GitHub.
Elaborare percorsi lunghi
Poiché la funzionalità di allineamento alle strade deduce la posizione in base al percorso completo, anziché ai singoli punti, devi prestare attenzione quando elabori percorsi lunghi (ovvero percorsi che superano il limite di 100 punti per richiesta).
Per trattare le singole richieste come un unico percorso lungo, devi includere una sovrapposizione in modo che i punti finali della richiesta precedente vengano inclusi come primi punti della richiesta successiva. Il numero di punti da includere dipende dall'accuratezza dei dati. Dovresti includere più punti per le richieste a bassa precisione.
Questo esempio utilizza il client Java per i servizi Google Maps per inviare richieste impaginate e poi unisce i dati, inclusi i punti interpolati, nell'elenco restituito.
/** * 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; }
Di seguito sono riportati i dati di cui sopra dopo l'esecuzione delle richieste di allineamento alle strade. La linea rossa rappresenta i dati non elaborati, mentre la linea blu rappresenta i dati allineati.

Utilizzare la quota in modo efficiente
La risposta a una richiesta di allineamento alle strade include un elenco di ID luogo
che corrispondono ai punti forniti, potenzialmente con punti aggiuntivi se hai
impostato interpolate=true.
Per utilizzare in modo efficiente la quota consentita per una richiesta di limiti di velocità, devi eseguire query solo per gli ID luogo univoci nella richiesta. Questo esempio utilizza il client Java per i servizi Google Maps per eseguire query sui limiti di velocità da un elenco di ID luogo.
/** * 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; }
Di seguito sono riportati i dati di cui sopra con i limiti di velocità contrassegnati in ogni ID luogo univoco.

Interazione con altre API
Uno dei vantaggi di avere gli ID luogo restituiti nelle risposte di allineamento alle strade è che puoi utilizzare l'ID luogo in molte delle API di Google Maps Platform. Questo esempio utilizza il client Java per i servizi Google Maps per eseguire il geocoding di un luogo restituito dalla richiesta di allineamento alle strade riportata sopra.
/** * 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; }
Qui l'indicatore del limite di velocità è stato annotato con l'indirizzo dell'API Geocoding.

Codice campione
Considerazioni
Il codice che supporta questo documento è disponibile come singola app per Android a scopo illustrativo. In pratica, non devi distribuire le chiavi API lato server in un'app per Android, poiché la chiave non può essere protetta dall'accesso non autorizzato di terze parti. Per proteggere le chiavi, devi invece eseguire il deployment del codice che utilizza l'API come proxy lato server e fare in modo che l'app per Android invii le richieste utilizzando il proxy, per assicurarti che le richieste siano autorizzate.
Scarica
Scarica il codice da GitHub.