Я получил/адаптировал следующий код из книги Адама Фримана "Metro раскрыл: создание приложений Windows 8 с помощью XAML и С#", чтобы получить адрес, когда известны координаты:
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");
}
Как я могу получить обратное (координаты, если адрес известен)
UPDATE
Я добавляю щедрость к этому; то, что у меня уже (показано выше), - это обратное геокодирование (получение адреса для координат); мне нужно геокодирование (получение координат для адреса).
Основываясь на моем обратном коде геокодирования выше, я предполагаю, что это может быть примерно так:
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 Риверсайд-авеню", я предполагаю.
Но для геокодирования (требуется два значения, долгота и широта) у меня есть этот псевдокод:
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;
... но даже с этим (близко, но без Боб Сегер), 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 (США)"
То, что я считаю странным в документации, заключается в том, что среди частей адреса нет элемента состояния. если это действительно так, а не только надзор за документацией, мой код выше завершится неудачно при вызове GetNamedString ( "состояние" ).
Я все еще не уверен, что правильный синтаксис и т.д. должен быть для противоположного (геокодирования) метода, получая координаты обратно после прохождения по адресу.
ОБНОВЛЕНИЕ 5
Хорошо, я скачал Json.NET и получил его компиляцию. Я еще не тестировал, но я пометил Питера Ричи как ответ (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);
}
также: На обратном пути к этому форуму произошло смешное: при установке Json.NET через NuGet я также увидел "самый быстрый JSON-сериализатор .NET от ServiceStack", который утверждает, что он на 3 раза быстрее, чем Json.NET. FIWW, он был обновлен совсем недавно, чем Json.NET. Мысли/реакция?
ОБНОВЛЕНИЕ 6
У меня есть этот код для реализации этого (идентификатор и код приложения были изменены для защиты полубесконечного (Я)):
// 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();
... Я получаю это сообщение err msg: "* System.InvalidOperationException не обрабатывается кодом пользователя HResult = -2146233079 Сообщение = Последовательность не содержит элементов Источник = System.Core Трассировки стека: в System.Linq.Enumerable.First [TSource] (источник IEnumerable`1) в SpaceOverlays.SOs_Classes.SOs_Utils.d__12.MoveNext() в c:... *"
Значение thisNetResponse:
{"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"}}}
... так что окажется, что там есть достоверная информация, например, должна возвращать "Хартфорд, Иллинойс"
И, во всяком случае, пустое возвращаемое значение не должно генерировать исключение, я бы подумал...