3

小さな地域の Open Street Map データをダウンロードします。データをフィルタリングして、特別なカテゴリのノードを取得したいと考えています。

これは OSM データのサンプルです

 <node id="505126369" lat="31.2933856" lon="34.2687443" user="JumpStart International" uid="125156" visible="true" version="1" changeset="2568758" timestamp="2009-09-22T13:05:10Z"/>
 <node id="505126372" lat="31.2682934" lon="34.2745680" user="JumpStart International" uid="125156" visible="true" version="1" changeset="2568758" timestamp="2009-09-22T13:05:10Z"/>
 <node id="505126375" lat="31.2953082" lon="34.3471630" user="JumpStart International" uid="125156" visible="true" version="1" changeset="2568758" timestamp="2009-09-22T13:05:10Z"/>
 <node id="505126378" lat="31.2807872" lon="34.2757999" user="JumpStart International" uid="125156" visible="true" version="1" changeset="2568758" timestamp="2009-09-22T13:05:11Z">
   <tag k="amenity" v="school"/>
   <tag k="name" v="Al Aqqad Basic &amp; Secondary Female School"/>
   <tag k="name:ar" v="مدرسة العقاد الأساسية والثانوية للبنات"/>
  </node>

学校全体、病院全体のデータを取得したい。

誰かが PHP や Java で XML 解析を行ったことがあれば、それを私とすべての興味のある人と共有していただければ幸いです。

編集 これは私が持っている簡単なスタートです

$dataFile = base_url() . 'media/files/osmdata/map_3.xml';
    //echo ($dataFile);

    $xml = simplexml_load_file($dataFile);

    //    $countTotal = count($xml->node);
    //   echo 'here'.$countTotal;
    foreach ($xml as $key => $val) {
        var_dump($val);
               // can't manage things overs here

    }
4

1 に答える 1

8

以下は、私がコンパイルした PHP SimpleXML を使用した OSM Overpass API の小さな例です。なぜなら、ここには PHP 用のものはありませんが、私は OSM が大好きなので、いくつかの有用な例を示しましょう。

最初の部分では、標準の PHP を使用して Overpass エンドポイントをクエリする方法を示します。すでにハードディスクにデータを保存しているため、その部分は必要ありません。

<?php
/**
 * OSM Overpass API with PHP SimpleXML / XPath
 *
 * PHP Version: 5.4 - Can be back-ported to 5.3 by using 5.3 Array-Syntax (not PHP 5.4's square brackets)
 */


//
// 1.) Query an OSM Overpass API Endpoint
//

$query = 'node
  ["amenity"~".*"]
  (38.415938460513274,16.06338500976562,39.52205163048525,17.51220703125);
out;';

$context = stream_context_create(['http' => [
    'method'  => 'POST',
    'header' => ['Content-Type: application/x-www-form-urlencoded'],
    'content' => 'data=' . urlencode($query),
]]);

# please do not stress this service, this example is for demonstration purposes only.
$endpoint = 'http://overpass-api.de/api/interpreter';
libxml_set_streams_context($context);
$start = microtime(true);

$result = simplexml_load_file($endpoint);
printf("Query returned %2\$d node(s) and took %1\$.5f seconds.\n\n", microtime(true) - $start, count($result->node));

あなたにとって、2番目の部分はより興味深いものです。それは、すでに持っている XML データをクエリすることです。これは xpath で最も簡単に実行できます。使用される PHP XML ライブラリは、XPath 1.0をサポートする libxml に基づいており、さまざまなクエリのニーズを非常によくカバーします。

次の例では、すべての学校を一覧表示し、その名前も取得しようとしています。私のサンプル データには翻訳がなかったので、翻訳についてはまだ説明していませんが、翻訳を含むあらゆる種類の名前を探して、特定の名前を優先することもできます):

//
// 2.) Work with the XML Result
//

# get all school nodes with xpath
$xpath = '//node[tag[@k = "amenity" and @v = "school"]]';
$schools = $result->xpath($xpath);
printf("%d School(s) found:\n", count($schools));
foreach ($schools as $index => $school)
{
    # Get the name of the school (if any), again with xpath
    list($name) = $school->xpath('tag[@k = "name"]/@v') + ['(unnamed)'];
    printf("#%02d: ID:%' -10s  [%s,%s]  %s\n", $index, $school['id'], $school['lat'], $school['lon'], $name);
}

ここでの重要なポイントは、xpath クエリです。2 つが使用され、最初の 1 つは特定のタグを持つノードを取得します。これはあなたにとって最も興味深いものだと思います:

//node[tag[@k = "amenity" and @v = "school"]]

この行は次のように述べています: k属性値"amenity"v属性値"school"を持つtag要素を持つすべてのnode要素を教えてください。これは、amenity school でタグ付けされたノードを除外する必要がある条件です。

さらに xpath が再び使用され、名前があるかどうかを確認し、名前がある場合は取得するために、これらの学校ノードに相対的になります。

tag[@k = "name"]/@v'

この行は次のように述べています: 現在のノードに関連して、 k属性値"name"としてタグ要素からv属性を与えてください。ご覧のとおり、いくつかの部分は以前のラインに再び似ています。どちらもニーズに合わせて採用できると思います。

すべての学校ノードに名前があるわけではないため、(空の) 結果配列に追加することで、表示用にデフォルトの文字列が提供されます。

list($name) = $school->xpath('tag[@k = "name"]/@v') + ['(unnamed)'];
                                                    ^^^^^^^^^^^^^^^
                                                Provide Default Value

したがって、そのコード例の結果は次のとおりです。

Query returned 907 node(s) and took 1.10735 seconds.
10 School(s) found:
#00: ID:332534486   [39.5017565,16.2721899]  Scuola Primaria
#01: ID:1428094278  [39.3320912,16.1862820]  (unnamed)
#02: ID:1822746784  [38.9075566,16.5776597]  (unnamed)
#03: ID:1822755951  [38.9120272,16.5713431]  (unnamed)
#04: ID:1903859699  [38.6830409,16.5522243]  Liceo Scientifico Statale A. Guarasci
#05: ID:2002566438  [39.1347698,16.0736924]  (unnamed)
#06: ID:2056891127  [39.4106679,16.8254844]  (unnamed)
#07: ID:2056892999  [39.4124687,16.8286119]  (unnamed)
#08: ID:2272010226  [39.4481717,16.2894353]  SCUOLA DELL'INFANZIA SAN FRANCESCO
#09: ID:2272017152  [39.4502366,16.2807664]  SCUOLA MEDIA 

これがすでに役立つことを願っています。さらに明確な質問がある場合はお知らせください。


(by rbwilkinson ): これは、他の値を見つけるために追加のパラメーターを追加する方法です。次の例では、1 キロメートル以内にある他のプロパティを検索します。

$query = 'node
  ["addr:postcode"~"RM12"]
  (51.5557914,0.2118915,51.5673083,0.2369398);
   node
  (around:1000)
  ["amenity"~"fast_food"];
           out;';

$context = stream_context_create(['http' => [
    'method'  => 'POST',
    'header' => ['Content-Type: application/x-www-form-urlencoded'],
    'content' => 'data=' . urlencode($query),
]]);

$endpoint = 'http://overpass-api.de/api/interpreter';
libxml_set_streams_context($context);

$result = simplexml_load_file($endpoint);
printf("Query returned %2\$d node(s) and took %1\$.5f seconds.\n\n", microtime(true) - $start, count($result->node));
}
于 2013-04-21T12:35:21.653 に答える