12

Adam Freemanの著書「MetroRevealed:Building Windows 8 apps with XAML and C#」から次のコードを派生/適応させて、座標がわかっているときにアドレスを取得しました。

public static async Task<string> GetAddressForCoordinates(double latitude, double longitude)
{
    HttpClient httpClient = new HttpClient {BaseAddress = new Uri("http://nominatim.openstreetmap.org")};
    HttpResponseMessage httpResult = await httpClient.GetAsync(
        String.Format("reverse?format=json&lat={0}&lon={1}", latitude, longitude));
JsonObject jsonObject = JsonObject.Parse(await httpResult.Content.ReadAsStringAsync());

return jsonObject.GetNamedObject("address").GetNamedString("road");

}

どうすれば反対の結果を得ることができますか(住所がわかっている場合は座標)?

アップデート

私はこれに賞金を追加しています。私がすでに持っているのは(上に表示されている)逆ジオコーディング(座標のアドレスを取得する)です。必要なのはジオコーディング(住所の座標を取得)です。

上記の逆ジオコーディングコードに基づいて、次のようになると思います。

public static async Task<string> GetCoordinatesForAddress(string address)
{
    HttpClient httpClient = new HttpClient {BaseAddress = new Uri("http://nominatim.openstreetmap.org")};
    HttpResponseMessage httpResult = await httpClient.GetAsync(
        String.Format("format=json&address={0}", address));

    JsonObject jsonObject = JsonObject.Parse(await httpResult.Content.ReadAsStringAsync());

    return jsonObject.GetNamedObject("address").GetNamedString("lat"); // <-- what about "lon"?
}

...しかし、2つの座標(経度と緯度)の値を組み合わせる方法がわかりません(これが正しい、またはほぼ正しいと仮定して)。誰かがこれを確認したり、クリーンアップしたり、より良い例を提供したりできますか(ノミナティムを使用するかどうかにかかわらず)?

更新2

以下のPeterRitchieの質問/コメントに答えるには:

オリジナル(逆ジオコーディングコード)には、次のものがあります。

return jsonObject.GetNamedObject("address").GetNamedString("road");

それは単に道を返すだけです。だから「157リバーサイドアベニュー」みたいなものだと思います。

しかし、ジオコーディング(経度と緯度の2つの値が必要)の場合、次の擬似コードがあります。

return jsonObject.GetNamedObject("address").GetNamedString("lat"); // <-- what about "lon"?

<stringしたがって、戻り値をTask >からTaskに変更して呼び出す必要があるかどうかはわかりません<List(冗長擬似コード)[注:文字列のリストを使用してTaskの山括弧をエスケープするのに苦労しています]:

var latitude jsonObject.GetNamedObject("address").GetNamedString("lat");
var longitude jsonObject.GetNamedObject("address").GetNamedString("lat");
List<string> listCoordinates = new List<string>();
listCoordinates.Add(latitude);
listCoordinates.Add(longitude);
return listCoordinates;

...またはそのように:

string latitude jsonObject.GetNamedObject("address").GetNamedString("lat");
string longtude jsonObject.GetNamedObject("address").GetNamedString("long");
return string.Format("{0};{1}", latitude, longitude);

...また ???

更新3

ジオコーディング用に提供されたJsonコードに応じて:

元の逆ジオコードコードに基づくと、呼び出しは次のようになりません。

HttpClient httpClient = new HttpClient { BaseAddress = new Uri("http://nominatim.openstreetmap.org/") };
var httpResult = await httpClient.GetAsync(
    String.Format("search?format=json&addressdetails={0}", address);

...しかしとにかく:JsonArrayは認識されますが、JArrayタイプは認識されません。JsonValueは認識されますが、JValueタイプは認識されません。JsonConverterタイプは認識されません。おそらくJson.Netの一部ですか?

提案されたコードをコンパイルするのに最も近いのは次のとおりです。

var result = await httpResult.Content.ReadAsStringAsync();
var r = (JsonArray)JsonConverter.DeserializeObject(result);//<-- JsonConvert[er] not recognized; part of Json.NET?
var latString = ((JsonValue)r[0]["lat"]).ValueType as string;
var longString = ((JsonValue)r[0]["lon"]).ValueType as string;

...しかし、これでも(ボブ・シーガーは近いが、ボブ・シーガーはいない)、JsonConvertとJsonConverterは認識されません。

更新4

http://wiki.openstreetmap.org/wiki/Nominatim#Searchのドキュメントをより協調的に調べた後、私の元の(逆ジオコード)方法は次のように優れていると思います。

public static async Task`<string`> GetAddressForCoordinates(double latitude, double longitude)
{
    HttpClient httpClient = new HttpClient {BaseAddress = new Uri("http://nominatim.openstreetmap.org/")};
    HttpResponseMessage httpResult = await httpClient.GetAsync(
        String.Format("reverse?format=json&lat={0}&lon={1}", latitude, longitude));

    JsonObject jsonObject = JsonObject.Parse(await httpResult.Content.ReadAsStringAsync());

    string house = jsonObject.GetNamedObject("addressparts").GetNamedString("house");
    string road = jsonObject.GetNamedObject("addressparts").GetNamedString("road");
    string city = jsonObject.GetNamedObject("addressparts").GetNamedString("city");
    string state = jsonObject.GetNamedObject("addressparts").GetNamedString("state");
    string postcode = jsonObject.GetNamedObject("addressparts").GetNamedString("postcode");
    string country = jsonObject.GetNamedObject("addressparts").GetNamedString("country");
    return string.Format("{0} {1}, {2}, {3} {4} ({5})", house, road, city, state, postcode, country);
}

これにより、渡された対応する座標引数に対して、「157 Riverside Avenue、Champaign、IL 55555(USA)」のようなものが返されます。

ドキュメントについて私が奇妙だと思うのは、アドレス部分の中に「状態」要素がないことです。それが本当に真実であり、ドキュメントの見落としだけではない場合、上記の私のコードはGetNamedString( "state")の呼び出しで失敗します。

アドレスを渡した後に座標を取得する、反対の(geocode)メソッドの正しい構文などがどうあるべきかはまだわかりません。

更新5

さて、Json.NETをダウンロードしてコンパイルしました。まだテストしていませんが、Peter RitchieをTHE(50ポイント)の回答としてマークしました。

これは私が使用しているコードです:

public static async Task<string> GetCoordinatesForAddress(string address)
{
    HttpClient httpClient = new HttpClient { BaseAddress = new Uri("http://nominatim.openstreetmap.org/") };
    HttpResponseMessage httpResult = await httpClient.GetAsync(
        String.Format("search?q={0}&format=json&addressdetails=1", Pluggify(address))); // In my Pluggify() method, I replace spaces with + and then lowercase it all

    var result = await httpResult.Content.ReadAsStringAsync();
    var r = (JArray)JsonConvert.DeserializeObject(result);
    var latString = ((JValue)r[0]["lat"]).Value as string;
    var longString = ((JValue)r[0]["lon"]).Value as string;
    return string.Format("{0};{1}", latString, longString);
}

また、このフォーラムに戻る途中で面白いことが起こりました。NuGetを介してJson.NETをインストールしているときに、Json.NETの3倍の速度であると主張する「.NETのServiceStackによる最速のJSONシリアライザー」も見ました。FIWW、Json.NETよりも最近更新されました。考え/反応?

更新6

私はこれを実装するためにこのコードを持っています(アプリIDとコードは半無実(私)を保護するために変更されました):

// If address has not been explicitly entered, try to suss it out:
                    address = textBoxAddress1.Text.Trim();
                    lat = textBoxLatitude1.Text.Trim();
                    lng = textBoxLongitude1.Text.Trim();
                    if (string.IsNullOrWhiteSpace(address))
                    {
                        address = await SOs_Classes.SOs_Utils.GetAddressForCoordinates(lat, lng);
                    }

. . .

        public async static Task<string> GetAddressForCoordinates(string latitude, string longitude)
        {
            string currentgeoLoc = string.Format("{0},{1}", latitude, longitude);
            string queryString = string.Empty;
            string nokiaAppID = "j;dsfj;fasdkdf";
            object nokiaAppCode = "-14-14-1-7-47-178-78-4";
            var hereNetUrl = string.Format(
                "http://demo.places.nlp.nokia.com/places/v1/discover/search?at={0}&q={1}&app_id={2}    
&app_code={3}&accept=application/json",
                    currentgeoLoc, queryString, nokiaAppID, nokiaAppCode);    
            // get data from HERE.net REST API
            var httpClient = new HttpClient();
            var hereNetResponse = await httpClient.GetStringAsync(hereNetUrl);    
            // deseralize JSON from Here.net 
            using (var tr = new StringReader(hereNetResponse))
            using (var jr = new JsonTextReader(tr))
            {
                var rootObjectResponse = new JsonSerializer    
().Deserialize<JsonDOTNetHelperClasses.RootObject>(jr);    
                var firstplace = rootObjectResponse.results.items.First();
                return HtmlUtilities.ConvertToText(firstplace.vicinity);
                // NOTE: There is also a title (such as "Donut Shop", "Fire stations", etc.?) and type (such as "residence" or "business", etc.?)
            }
        }

...しかし、GetAddressForCoordinates()のこの行では:

        var firstplace = rootObjectResponse.results.items.First();

...このエラーメッセージが表示されます:"*System.InvalidOperationExceptionはユーザーコードによって処理されませんでしたHResult=-2146233079 Message=Sequenceには要素が含まれていませんSource=System.Core StackTrace:at System.Linq.Enumerable.First [TSource](IEnumerable` 1ソース)SpaceOverlays.SOs_Classes.SOs_Utils.d__12.MoveNext()in c:... * "

hereNetResponseの値は次のとおりです。

{"results":{"items":[]},"search":{"context":{"location":{"position":[38.804967,-90.113183],"address":
{"postalCode":"62048","city":"Hartford","stateCode":"IL","county":"Madison","countryCode":"USA","country":"
USA","text":"Hartford IL 62048
USA"}},"type":"urn:nlp-types:place","href":"http://demo.places.nlp.nokia.com/places/v1/places/loc-
dmVyc2lvbj0xO3RpdGxlPUhhcnRmb3JkO2xhdD0zOC44MDQ5Njc7bG9uPS05MC4xMTMxODM7Y2l0eT1IY
XJ0Zm9yZDtwb3N0YWxDb2RlPTYyMDQ4O2NvdW50cnk9VVNBO3N0YXRlQ29kZT1JTDtjb3VudHk9TWFka
XNvbjtjYXRlZ29yeUlkPWNpdHktdG93bi12aWxsYWdl;context=Zmxvdy1pZD02YmUzZDM4Yi0wNGVhLTUyM
jgtOWZmNy1kNWNkZGM0ODI5OThfMTM1NzQyMDI1NTg1M18wXzE2MA?
app_id=F6zpNc3TjnkiCLwl_Xmh&app_code=QoAM_5BaVDZvkE2jRvc0mw"}}}

...そのため、「ハートフォード、イリノイ」を返すなど、有効な情報が内部にあるように見えます。

とにかく、空白の戻り値は例外をスローするべきではないと思います...

4

3 に答える 3

17

あなたが求めているのは、単に「ジオコーディング」です。Nominatimを具体的に使用したい場合は、これを「検索」と呼びます。これは、ある程度、アドレス検証です。ただし、「検証」の一部には座標が含まれます(境界ボックス、緯度/経度など。検索対象と結果のタイプによって異なります)。結果については多くの詳細があり、ここに投稿するには多すぎます。ただし、この詳細については、http://wiki.openstreetmap.org/wiki/Nominatim#Search(例を含む)を参照してください。

結果(XML、JSON、またはHTML)を解析して、関心のあるフィールドを取得する必要があります。

アップデート1:

実際の値をどうするかについて:それは異なります。フォームで座標を表示したい場合は、lat文字列とlong文字列を個々のコントロールに配置するだけです。単一のコントロールに入れたい場合は、を使用できますstring.Format("{0}, {1}", latString, longString)Microsoft.Maps.MapControl.LocationWindowsストアアプリのさまざまなメソッド/タイプで座標を使用する場合は、クラスを使用する必要がある場合があります。例えば:

  Double latNumber;
  Double longNumber;
  if(false == Double.TryParse(latString, out latNumber)) throw new InvalidOperationException();
  if(false == Double.TryParse(longString, out longNumber)) throw new InvalidOperationException();
  var location = new Location(latNumber, longNumber);

上記は、応答からlatとlongを抽出しlatStringlongStringそれぞれに配置したことを前提としています。

一部のインターフェースでは、個別のdouble値としてlat / longが必要になる場合があります。その場合は、上記を使用latNumberlongNumberてください。

その上、それは本当にあなたが使いたいインターフェースに特に依存します。しかし、上記はほとんどのインターフェースを使用するのに十分なはずです。

アップデート2:

質問が「座標を取得する方法」ではなく「jsonオブジェクトを解析する方法」である場合は、JSon.Netを使用してjson結果の緯度/経度の文字列を取得することをお勧めします。例えば:

    var httpClient = new HttpClient();
    var httpResult = await httpClient.GetAsync(
        "http://nominatim.openstreetmap.org/search?q=135+pilkington+avenue,+birmingham&format=json&polygon=1&addressdetails=1");

    var result = await httpResult.Content.ReadAsStringAsync();
    var r = (JArray) JsonConvert.DeserializeObject(result);
    var latString = ((JValue) r[0]["lat"]).Value as string;
    var longString = ((JValue)r[0]["lon"]).Value as string;

...上記を参照してlatStringlongString

于 2012-12-23T17:05:09.343 に答える
3

グーグルマップのジオコーディングをお探しの場合は、
https ://developers.google.com/maps/documentation/geocoding/で見つけることができます。

使用例は次のとおりです。http:
//maps.googleapis.com/maps/api/geocode/json?address = 1600 + Amphitheatre + Parkway 、+ Mountain + View、+ CA&sensor = false

Jsonに変換し、Geometryオブジェクトを取得します。

基本的に、関数から2つの値を返す方法に関しては、C#では4つのオプションがあります。

  1. 両方のオブジェクトを同じタイプの単一のオブジェクトに変換し(文字列の場合は、行ったように区切り文字で区切るだけです)、後で両方を解析します。
  2. リストを返します-この場合はタスク(または配列)です。
  3. 両方を含む新しいクラス/オブジェクトを作成します(この場合、たとえばGeometryと呼ぶことができます)。
  4. 関数への参照としてオブジェクトを渡すように要求することにより、オブジェクトの一方または両方を返します。非同期関数では、これは非常に注意が必要ですが、誰が関数を呼び出し、誰が結果を処理するかによっては、それが可能になる場合があります。
于 2012-12-30T07:12:43.420 に答える
2

Microsoft Mappointには、使用できるAPIも含まれています。地理座標を返すことができます。

于 2012-12-23T17:32:27.977 に答える