المفاهيم المتقدمة

الحصول على البيانات

هناك العديد من الطرق للحصول على بيانات الموقع الجغرافي التي تم جمعها. نوضّح هنا أسلوبَين للحصول على البيانات لاستخدامهما مع ميزة المحاذاة إلى الطرق في 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 Client for Google Maps Services لإرسال الطلبات المقسّمة على صفحات، ثم ينضم مجددًا البيانات، بما في ذلك النقاط المُدخلة، إلى القائمة المعروضة.

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

في ما يلي البيانات الواردة أعلاه مع حدود السرعة الموضوعة لكل رقم تعريف مكان فريد.

لافتات حدود السرعة على الخريطة

التفاعل مع واجهات برمجة التطبيقات الأخرى

تتمثل إحدى فوائد عرض معرّفات الأماكن في ردود المحاذاة إلى الطرق في أنّه يمكنك استخدام معرّف المكان في العديد من واجهات برمجة تطبيقات "منصة خرائط Google". يستخدم هذا المثال برنامج 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;
}

تمت هنا إضافة تعليق توضيحي إلى علامة حد السرعة بالعنوان من واجهة برمجة تطبيقات الترميز الجغرافي.

عنوان مشفّر جغرافيًا يظهر على علامة

نموذج التعليمات البرمجية

الاعتبارات

يتوفّر الرمز البرمجي الذي تدعم هذه المقالة كتطبيق Android واحد لأغراض التوضيح. من الناحية العملية، يجب ألّا توزِّع مفاتيح واجهة برمجة التطبيقات من جهة الخادم في أحد تطبيقات Android لأنّ مفتاحك لا يمكن حمايته من الوصول غير المصرَّح به من جهة خارجية. بدلاً من ذلك، لتأمين مفاتيحك، يجب نشر الرمز الموجّه لواجهة برمجة التطبيقات كخادم وكيل من جهة الخادم وجعل تطبيق Android يرسل الطلبات عبر الخادم الوكيل، مع ضمان السماح بالطلبات.

تنزيل

نزِّل الرمز من GitHub.