קונספטים מתקדמים

הוספת נתונים

יש הרבה דרכים לקבל נתוני מיקום שנאספו. במאמר הזה מתוארות שתי טכניקות לאיסוף נתונים לשימוש בתכונה הצמדה לכבישים של Roads API.

GPX

‫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 גולמיים שנטענו במפה.

נתוני GPX גולמיים במפה

שירותי המיקום של Android

הדרך הטובה ביותר לאיסוף נתוני GPS ממכשיר Android משתנה בהתאם לתרחיש השימוש. כדאי לעיין בשיעור ההדרכה בנושא Android בנושא קבלת עדכוני מיקום, וגם בדוגמאות למיקום ב-Google Play ב-GitHub.

עיבוד נתיבים ארוכים

התכונה הצמדה לכבישים מסיקה את המיקום על סמך הנתיב המלא, ולא על סמך נקודות בודדות. לכן, צריך להיזהר כשמעבדים נתיבים ארוכים (כלומר, נתיבים שעוברים את המגבלה של 100 נקודות לכל בקשה).

כדי להתייחס לבקשות הנפרדות כאל נתיב ארוך אחד, צריך לכלול חפיפה מסוימת, כך שהנקודות האחרונות מהבקשה הקודמת ייכללו כנקודות הראשונות בבקשה הבאה. מספר הנקודות שצריך לכלול תלוי בדיוק של הנתונים. כדאי לכלול יותר נקודות בבקשות עם רמת דיוק נמוכה.

בדוגמה הזו נעשה שימוש בלקוח Java לשירותי מפות Google כדי לשלוח בקשות עם דפדוף, ואז הנתונים, כולל נקודות שחושבו באמצעות אינטרפולציה, מצורפים מחדש לרשימה שמוחזרת.

/**
 * 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;
}

אלה הנתונים מהטבלה שלמעלה אחרי הפעלת הבקשות להתאמה לכבישים. הקו האדום הוא הנתונים הגולמיים והקו הכחול הוא הנתונים המותאמים.

דוגמה לנתונים שהוצמדו לכבישים

שימוש יעיל במכסה

התשובה לבקשה של הצמדה לכבישים כוללת רשימה של מזהי מקומות שממופים לנקודות שסיפקתם, ויכול להיות שיהיו בה נקודות נוספות אם הגדרתם את interpolate=true.

כדי להשתמש ביעילות במכסת הבקשות המותרת שלכם לבקשת הגבלות מהירות, כדאי לשלוח בבקשה רק מזהי מקומות ייחודיים. בדוגמה הזו נעשה שימוש בלקוח Java לשירותי מפות Google כדי לשלוח שאילתה לגבי הגבלות מהירות מרשימה של מזהי מקומות.

/**
 * 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 אחרים

אחד היתרונות של קבלת מזהי מקומות בתשובות של הצמדה לכבישים הוא שאפשר להשתמש במזהה המקום בממשקי API רבים של Google Maps Platform. בדוגמה הזו נעשה שימוש בלקוח Java לשירותי מפות Google כדי לבצע קידוד גיאוגרפי של מקום שהוחזר מהבקשה שלמעלה להצמדת נקודה לכביש.

/**
 * 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;
}

כאן, סומן תמרור מגבלת המהירות עם הכתובת מ-Geocoding API.

כתובת עם קידוד גיאוגרפי שמוצגת בסמן

קוד לדוגמה

לתשומת ליבכם

הקוד שתומך במסמך הזה זמין כאפליקציית Android אחת להמחשה. בפועל, לא מומלץ להפיץ את מפתחות ה-API בצד השרת באפליקציית Android, כי אי אפשר להגן על המפתח מפני גישה לא מורשית מצד שלישי. במקום זאת, כדי לאבטח את המפתחות, צריך לפרוס את הקוד שפונה ל-API כפרוקסי בצד השרת, ולגרום לאפליקציית Android לשלוח בקשות באמצעות הפרוקסי, כדי לוודא שהבקשות מורשות.

הורדה

מורידים את הקוד מ-GitHub.