12

我从 Adam Freeman 的书“Metro Revealed:使用 XAML 和 C# 构建 Windows 8 应用程序”中派生/改编了以下代码,以便在已知坐标时获取地址:

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"?
}

...但我不知道如何组合两个坐标(经度和纬度)值(假设这是正确的,或者接近正确)。任何人都可以验证、清理或提供更好的示例(使用 nominatim 或其他方式)吗?

更新 2

要在下面回答 Peter Ritchie 的问题/评论:

在原始(反向地理编码代码)中,我有:

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

它只是返回道路;所以我假设像“157 Riverside Avenue”之类的东西。

但是对于地理编码(需要两个值,一个经度和一个纬度),我有这个伪代码:

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

所以我不知道是否需要将返回值从 Task <string> 更改为 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);

...但无论如何: JArray 类型无法识别,尽管 JsonArray 是。JValue 类型无法识别,但 JsonValue 可以。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;

...但即使这样(关闭但没有 Bob Seger),也无法识别 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);
}

对于传入的相应坐标 args,这将返回类似于:“ 157 Riverside Avenue, Champaign, IL 55555 (USA)

我对文档感到奇怪的是地址部分中没有“状态”元素。如果这是真的,而不仅仅是文档疏忽,我上面的代码将在调用 GetNamedString("state") 时失败。

我仍然不确定相反的(地理编码)方法应该是什么正确的语法等,在传入地址后获取坐标。

更新 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 时,我还看到了“.NET 的 ServiceStack 的最快 JSON 序列化程序”,它声称比 Json.NET 快 3 倍。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 处理消息=序列不包含元素 Source=System.Core StackTrace:在 System.Linq.Enumerable.First[TSource](IEnumerable` 1 来源)在 SpaceOverlays.SOs_Classes.SOs_Utils.d__12.MoveNext() 在 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:

至于如何处理实际值:这取决于。如果您想在表单中查看坐标,您可以简单地将经纬度字符串放入单独的控件中。如果你想把它放在一个单独的控件中,可以使用string.Format("{0}, {1}", latString, longString). 如果您想为 Windows 应用商店应用程序使用各种方法/类型的坐标,您可能需要使用Microsoft.Maps.MapControl.Location该类。例如:

  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 并将它们latString分别放入 中longString

某些接口可能需要 lat/long 作为单独的双精度值,在这种情况下只需使用latNumberlongNumber以上。

除此之外,它实际上具体取决于您要使用的接口。但是,以上内容应该足以让您使用大多数接口。

更新 2:

如果问题不是“如何获取坐标”而是“如何解析 json 对象”,那么我建议使用 JSon.Net 来获取 json 结果中的 lat/long 字符串。例如:

    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 对象。

关于如何从一个函数中返回两个值,在 C# 中你有 4 个选项:

  1. 将两个对象转换为相同类型的单个对象(在字符串的情况下,您只需将它们用分隔符分隔,就像您所做的那样),然后将它们解析出来。
  2. 返回一个列表 - 在这种情况下是一个任务(或数组)。
  3. 创建一个包含两者的新类/对象(例如,在这种情况下,您可以将其称为 Geometry)。
  4. 通过要求将它们作为对函数的引用传递来返回一个或两个对象。在异步函数中,这肯定会非常棘手,但取决于谁调用函数以及谁处理结果,这可能是可能的。
于 2012-12-30T07:12:43.420 回答
2

Microsoft Mappoint 还包含一个您可以使用的 API。它能够返回地理坐标。

于 2012-12-23T17:32:27.977 回答