Location服务之Geocoder (转载liuhe688)
2012-07-13 17:17
225 查看
提到Android基于位置的服务,就不得不提android.location包,location包提供了很便捷的API来实现基于位置的服务。主要包括Geocoder和LocationManager。今天就先来介绍一下Geocoder。
Geocoder可以在街道地址和经纬度地图坐标之间进行转换。它提供了对两种地理编码功能的访问:
Forward Geocoding(前向地理编码):查找某个地址的经纬度
Reverse Geocoding(反向地理编码):查找一个给定的经纬度所对应的街道地址。
分别对应以下方法:
[java]
view plaincopy
List<Address> getFromLocationName(String locationName, int maxResults)
List<Address> getFromLocation(double latitude,double longitude,int maxResults);
我们新建一个location的项目。因为示例要使用到地图服务,所以创建时Build Target要选择Google APIs这一项。
然后修改/res/layout/main.xml,代码如下:
[xhtml]
view plaincopy
<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="fill_parent"
android:layout_height="fill_parent">
<com.google.android.maps.MapView
android:id="@+id/mapView"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:clickable="true"
android:apiKey="your apiKey goes here"/>
<LinearLayout
android:layout_width="fill_parent"
android:layout_height="wrap_content">
<EditText
android:id="@+id/name"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_weight="1"/>
<Button
android:id="@+id/find"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Find"/>
</LinearLayout>
</FrameLayout>
然后我们来看一下MainActivity.java文件,代码如下:
[java]
view plaincopy
package com.scott.location;
import java.io.IOException;
import java.util.List;
import android.location.Address;
import android.location.Geocoder;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.ImageView;
import android.widget.Toast;
import com.google.android.maps.GeoPoint;
import com.google.android.maps.MapActivity;
import com.google.android.maps.MapView;
import com.google.android.maps.MapView.LayoutParams;
public class MainActivity extends MapActivity {
private MapView mapView;
private EditText name;
private Button find;
private Geocoder geocoder;
private static final double lat = 39.908716;
private static final double lng = 116.397529;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
mapView = (MapView) findViewById(R.id.mapView);
mapView.getController().setZoom(17);
mapView.getController().animateTo(new GeoPoint((int)(lat * 1E6),(int)(lng * 1E6)));
geocoder = new Geocoder(this);
name = (EditText) findViewById(R.id.name);
find = (Button) findViewById(R.id.find);
find.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
String keyword = name.getText().toString();
try {
List<Address> addrs = geocoder.getFromLocationName(keyword, 3);
if (addrs != null && addrs.size() > 0) {
int latE6 = (int) (addrs.get(0).getLatitude() * 1E6);
int lngE6 = (int) (addrs.get(0).getLongitude() * 1E6);
GeoPoint point = new GeoPoint(latE6, lngE6);
mapView.getController().animateTo(point);
final MapView.LayoutParams params = new MapView.LayoutParams(LayoutParams.WRAP_CONTENT,
LayoutParams.WRAP_CONTENT, point, LayoutParams.BOTTOM_CENTER);
final ImageView marker = new ImageView(MainActivity.this);
marker.setImageResource(R.drawable.marker);
marker.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Toast.makeText(getApplicationContext(), "hello geocoder!", Toast.LENGTH_SHORT).show();
}
});
mapView.addView(marker, params);
}
} catch (IOException e) {
e.printStackTrace();
}
}
});
}
@Override
protected boolean isRouteDisplayed() {
return false;
}
}
最后需要在AndroidManifest.xml中的application标签之间加入google map library:
[xhtml]
view plaincopy
<uses-library android:name="com.google.android.maps" />
然后就是一些位置服务所需的权限:
[xhtml]
view plaincopy
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
有一点需要跟朋友们提一下,Geocoder查找是同步地进行的,因此,它们将会阻塞调用它们的线程。对于低速的数据连接来说,这可能会导致出现ANR(Application Not Respond)的问题。在大部分情况下,更好的做法是把这些查找移动到服务或者后台线程中,前面几篇文章中也涉及到了一些异步任务的相关知识,不熟悉的朋友们可以参看以下Handler和AsyncTask的使用。为了清晰和简洁起见,上面代码中的查找操作直接放在了UI线程中,应用时最好不要这样写。
做完上面的操作,基本上就算是完成了该示例。不过如果使用模拟器时用的是Google API v8时会出现一个异常,将无法完成查找功能:
这是什么原因呢?在网上搜寻了一通,发现讨论区,真的很给力:http://code.google.com/p/android/issues/detail?id=8816
看看个别的评论:
看来真机和v7都没问题,v8会出问题。
也有高人给出解决方案,另辟蹊径,使用访问url的方式,获取json数据,然后解析获取经度和纬度,最后再组装成一个GeoPoint对象即可。我们新建一个LocationUtil.java文件,代码如下:
[java]
view plaincopy
package com.scott.location;
import java.io.InputStream;
import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.client.HttpClient;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.impl.client.DefaultHttpClient;
import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;
import com.google.android.maps.GeoPoint;
public class LocationUtil {
public static GeoPoint getFromLocationName(String address) {
String url = "http://maps.google.com/maps/api/geocode/json";
HttpGet httpGet = new HttpGet(url + "?sensor=false&address=" + address);
HttpClient client = new DefaultHttpClient();
HttpResponse response;
StringBuilder stringBuilder = new StringBuilder();
try {
response = client.execute(httpGet);
HttpEntity entity = response.getEntity();
InputStream stream = entity.getContent();
int b;
while ((b = stream.read()) != -1) {
stringBuilder.append((char) b);
}
} catch (Exception e) {
e.printStackTrace();
}
JSONObject jsonObject = new JSONObject();
try {
jsonObject = new JSONObject(stringBuilder.toString());
} catch (JSONException e) {
e.printStackTrace();
}
return getGeoPoint(jsonObject);
}
private static GeoPoint getGeoPoint(JSONObject jsonObject) {
try {
JSONArray array = (JSONArray) jsonObject.get("results");
JSONObject first = array.getJSONObject(0);
JSONObject geometry = first.getJSONObject("geometry");
JSONObject location = geometry.getJSONObject("location");
double lat = location.getDouble("lat");
double lng = location.getDouble("lng");
return new GeoPoint((int) (lat * 1E6), (int) (lng * 1E6));
} catch (JSONException e) {
e.printStackTrace();
}
return null;
}
}
然后对MainActivity.java关键地方进行改写:
[java]
view plaincopy
GeoPoint point = LocationUtil.getFromLocationName(keyword);
大功告成,当我们搜索后把结果中的第一个显示到地图中,然后点击图标时,弹出提示。我们来看一下效果:
以上是关于android.location包基于位置服务中Geocoder的基本介绍,更多的内容会在以后找机会和大家再分享。
Geocoder可以在街道地址和经纬度地图坐标之间进行转换。它提供了对两种地理编码功能的访问:
Forward Geocoding(前向地理编码):查找某个地址的经纬度
Reverse Geocoding(反向地理编码):查找一个给定的经纬度所对应的街道地址。
分别对应以下方法:
[java]
view plaincopy
List<Address> getFromLocationName(String locationName, int maxResults)
List<Address> getFromLocation(double latitude,double longitude,int maxResults);
我们新建一个location的项目。因为示例要使用到地图服务,所以创建时Build Target要选择Google APIs这一项。
然后修改/res/layout/main.xml,代码如下:
[xhtml]
view plaincopy
<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="fill_parent"
android:layout_height="fill_parent">
<com.google.android.maps.MapView
android:id="@+id/mapView"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:clickable="true"
android:apiKey="your apiKey goes here"/>
<LinearLayout
android:layout_width="fill_parent"
android:layout_height="wrap_content">
<EditText
android:id="@+id/name"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_weight="1"/>
<Button
android:id="@+id/find"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="Find"/>
</LinearLayout>
</FrameLayout>
然后我们来看一下MainActivity.java文件,代码如下:
[java]
view plaincopy
package com.scott.location;
import java.io.IOException;
import java.util.List;
import android.location.Address;
import android.location.Geocoder;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import android.widget.ImageView;
import android.widget.Toast;
import com.google.android.maps.GeoPoint;
import com.google.android.maps.MapActivity;
import com.google.android.maps.MapView;
import com.google.android.maps.MapView.LayoutParams;
public class MainActivity extends MapActivity {
private MapView mapView;
private EditText name;
private Button find;
private Geocoder geocoder;
private static final double lat = 39.908716;
private static final double lng = 116.397529;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
mapView = (MapView) findViewById(R.id.mapView);
mapView.getController().setZoom(17);
mapView.getController().animateTo(new GeoPoint((int)(lat * 1E6),(int)(lng * 1E6)));
geocoder = new Geocoder(this);
name = (EditText) findViewById(R.id.name);
find = (Button) findViewById(R.id.find);
find.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
String keyword = name.getText().toString();
try {
List<Address> addrs = geocoder.getFromLocationName(keyword, 3);
if (addrs != null && addrs.size() > 0) {
int latE6 = (int) (addrs.get(0).getLatitude() * 1E6);
int lngE6 = (int) (addrs.get(0).getLongitude() * 1E6);
GeoPoint point = new GeoPoint(latE6, lngE6);
mapView.getController().animateTo(point);
final MapView.LayoutParams params = new MapView.LayoutParams(LayoutParams.WRAP_CONTENT,
LayoutParams.WRAP_CONTENT, point, LayoutParams.BOTTOM_CENTER);
final ImageView marker = new ImageView(MainActivity.this);
marker.setImageResource(R.drawable.marker);
marker.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Toast.makeText(getApplicationContext(), "hello geocoder!", Toast.LENGTH_SHORT).show();
}
});
mapView.addView(marker, params);
}
} catch (IOException e) {
e.printStackTrace();
}
}
});
}
@Override
protected boolean isRouteDisplayed() {
return false;
}
}
最后需要在AndroidManifest.xml中的application标签之间加入google map library:
[xhtml]
view plaincopy
<uses-library android:name="com.google.android.maps" />
然后就是一些位置服务所需的权限:
[xhtml]
view plaincopy
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
有一点需要跟朋友们提一下,Geocoder查找是同步地进行的,因此,它们将会阻塞调用它们的线程。对于低速的数据连接来说,这可能会导致出现ANR(Application Not Respond)的问题。在大部分情况下,更好的做法是把这些查找移动到服务或者后台线程中,前面几篇文章中也涉及到了一些异步任务的相关知识,不熟悉的朋友们可以参看以下Handler和AsyncTask的使用。为了清晰和简洁起见,上面代码中的查找操作直接放在了UI线程中,应用时最好不要这样写。
做完上面的操作,基本上就算是完成了该示例。不过如果使用模拟器时用的是Google API v8时会出现一个异常,将无法完成查找功能:
这是什么原因呢?在网上搜寻了一通,发现讨论区,真的很给力:http://code.google.com/p/android/issues/detail?id=8816
看看个别的评论:
看来真机和v7都没问题,v8会出问题。
也有高人给出解决方案,另辟蹊径,使用访问url的方式,获取json数据,然后解析获取经度和纬度,最后再组装成一个GeoPoint对象即可。我们新建一个LocationUtil.java文件,代码如下:
[java]
view plaincopy
package com.scott.location;
import java.io.InputStream;
import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.client.HttpClient;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.impl.client.DefaultHttpClient;
import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;
import com.google.android.maps.GeoPoint;
public class LocationUtil {
public static GeoPoint getFromLocationName(String address) {
String url = "http://maps.google.com/maps/api/geocode/json";
HttpGet httpGet = new HttpGet(url + "?sensor=false&address=" + address);
HttpClient client = new DefaultHttpClient();
HttpResponse response;
StringBuilder stringBuilder = new StringBuilder();
try {
response = client.execute(httpGet);
HttpEntity entity = response.getEntity();
InputStream stream = entity.getContent();
int b;
while ((b = stream.read()) != -1) {
stringBuilder.append((char) b);
}
} catch (Exception e) {
e.printStackTrace();
}
JSONObject jsonObject = new JSONObject();
try {
jsonObject = new JSONObject(stringBuilder.toString());
} catch (JSONException e) {
e.printStackTrace();
}
return getGeoPoint(jsonObject);
}
private static GeoPoint getGeoPoint(JSONObject jsonObject) {
try {
JSONArray array = (JSONArray) jsonObject.get("results");
JSONObject first = array.getJSONObject(0);
JSONObject geometry = first.getJSONObject("geometry");
JSONObject location = geometry.getJSONObject("location");
double lat = location.getDouble("lat");
double lng = location.getDouble("lng");
return new GeoPoint((int) (lat * 1E6), (int) (lng * 1E6));
} catch (JSONException e) {
e.printStackTrace();
}
return null;
}
}
然后对MainActivity.java关键地方进行改写:
[java]
view plaincopy
GeoPoint point = LocationUtil.getFromLocationName(keyword);
大功告成,当我们搜索后把结果中的第一个显示到地图中,然后点击图标时,弹出提示。我们来看一下效果:
以上是关于android.location包基于位置服务中Geocoder的基本介绍,更多的内容会在以后找机会和大家再分享。
相关文章推荐
- Location服务之LocationManager (转载liuhe688)
- Location---Location服务之Geocoder
- Android 地理位置Location服务之LocationManager,Geocoder详解
- GoogleMaps Location服务之Geocoder
- Android Location服务之Geocoder
- Location服务之Geocoder
- Location服务之Geocoder
- Android 地理位置Location服务之LocationManager,Geocoder详解
- Location服务之Geocoder
- Location服务之Geocoder和LocationManager
- Location服务之Geocoder
- 使用 HttpWebRequest 类调用 WEB 服务的示例(C#)【转载】
- 用window.location.href实现刷新另个框架页面 (转载自 仰天一笑)
- Linux设置服务自启动(转载)
- 基础总结篇之七:ContentProvider之读写短消息 (转载liuhe688)
- 修改Java文件不用重启Tomcat服务----热部署、热加载(转载)
- 转载:交叉编译inetutils并配置telnet服务
- [转载]微服务实践(五):微服务的事件驱动数据管理
- ORA-12514 TNS 监听程序当前无法识别连接描述符中请求服务 的解决方法【转载】
- 在android用Google Geocoding API服务解析地址(关键词:android/GPS/Geocoding API/getLocationFrom()返回null)