Как получить координаты, когда адрес известен?
я получил / адаптировал следующий код из книги Адама Фримена " Metro Revealed: 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"?
}
...но я не знаю, как объединить два значения координат (долгота и широта) (предполагая, что это правильно или близко к правильности). Может ли кто-нибудь проверить это, очистить его или предоставить лучший пример (используя nominatim или иначе)?
обновление 2
ответить Петр Вопрос/комментарий Ричи ниже:
в оригинале (обратный код геокодирования), у меня есть:
return jsonObject.GetNamedObject("address").GetNamedString("road");
он просто возвращает дорогу; поэтому я предполагаю, что-то вроде "157 Riverside Avenue".
но для геокодирования (нужны два значения, долгота и широта) у меня есть этот псевдокод:
return jsonObject.GetNamedObject("address").GetNamedString("lat"); // <-- what about "lon"?
поэтому я не знаю, нужно ли мне изменить возвращаемое значение из Task<string
> задачи<List
и вызов (подробный псевдокод) [Примечание: у меня есть трудно избежать угловых скобок для задачи со списком строк]:
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?
ближайший я могу прийти, чтобы получить код proferred компиляции:
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);
}
это вернулось бы, для соответствующих координат, переданных args, что-то вроде: "157 Riverside Avenue, Champaign, IL 55555 (USA)"
что я нахожу странным в документации, так это отсутствие элемента "состояние" среди адресных частей; если это действительно так, а не просто контроль документации, мой код выше потерпит неудачу при вызове 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();
...Я вам это ошибка msg: "* System.Исключение InvalidOperationException не обработано пользовательским кодом Значение HRESULT=-2146233079 Message=последовательность не содержит элементов исходная система.Ядро Трассировка стека: в системе.В LINQ.Перечислимый.Первый [TSource] (IEnumerable ' 1 Источник) в SpaceOverlays.SOS-классы.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"}}}
...так что, похоже, там есть действительная информация, например, должна вернуться " Хартфорд, IL"
и, во всяком случае, пустое возвращаемое значение не должно вызывать исключения, я бы подумал...
3 ответов
то, о чем вы спрашиваете, просто "геокодирование". Если вы хотите использовать Nominatim конкретно, они называют это "поиск". Это, в определенной степени, проверка адреса; но часть "проверки" включает координаты (ограничивающая рамка, lat/long и т. д.; в зависимости от того, что ищется и какой тип результата). Есть много деталей о результатах, слишком много, чтобы просто опубликовать здесь; но эту деталь можно найти здесь: http://wiki.openstreetmap.org/wiki/Nominatim#Search (включая examles).
вам придется проанализировать результаты (XML, JSON или HTML), чтобы получить поля, которые вас интересуют.
обновление 1:
а что делать с фактическими значениями: это зависит. Если вы хотите просмотреть координаты в форме, вы можете просто поместить lat и длинные строки в отдельные элементы управления. Если вы хотите поместить его в один элемент управления, можно использовать string.Format("{0}, {1}", latString, longString)
. Если вы хотите использовать coords с различными методами / типами для приложения Магазина 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 как отдельные двойные значения, в этом случае просто используйте latNumber
и longNumber
выше.
сверх того, это действительно зависит конкретно от интерфейсы, которые вы хотите использовать. Но вышеизложенное должно дать вам достаточно, чтобы использовать большинство интерфейсов.
обновление 2:
если вопрос не "как получить координаты", а "как анализировать объекты json", то я бы рекомендовал использовать JSon.Net чтобы добраться до строк lat/long в результате 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;
...см. выше w.r.т. что делать с latString
и longString
Если вы ищете Google maps geocoding, вы можете найти его здесь:
https://developers.google.com/maps/documentation/geocoding/
и пример использования is:
http://maps.googleapis.com/maps/api/geocode/json?address=1600+Amphitheatre+Parkway,+Mountain+View,+CA&sensor=false
преобразовать в Json и взять объект геометрии.
в отношении того, как, в принципе, вернуть два значения из функции в C# есть 4 варианта:
- преобразуйте оба объекта в один объект одного и того же типа (в случае строк вы просто отделяете их разделителем, как вы это сделали), а затем разбираете оба впоследствии.
- return a list - в этом случае задача (или массив).
- создайте новый класс/объект, содержащий оба (в этом случае вы можете назвать его геометрией, например).
- верните один или оба объекта, потребовав, чтобы они были переданы в качестве ссылки на функция. В асинхронной функции это должно быть очень сложно, но в зависимости от того, кто вызывает функцию и кто обрабатывает результат, это может быть возможно.
Майкрософт веб-службы mappoint также contaisn в API, который вы могли бы использовать. Он способен возвращать геокоординаты.