提到Android基于位置的服务,就不得不提android.location包,location包提供了很便捷的API来实现基于位置的服务。主要包括Geocoder和LocationManager。今天就先来介绍一下Geocoder。
Geocoder可以在街道地址和经纬度地图坐标之间进行转换。它提供了对两种地理编码功能的访问:
Forward Geocoding(前向地理编码):查找某个地址的经纬度
Reverse Geocoding(反向地理编码):查找一个给定的经纬度所对应的街道地址。
分别对应以下方法:
- List getFromLocationName(String locationName, int maxResults)
- List getFromLocation(double latitude,double longitude,int maxResults);
我们新建一个location的项目。因为示例要使用到地图服务,所以创建时Build Target要选择Google APIs这一项。
然后修改/res/layout/main.xml,代码如下:
- 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文件,代码如下:
- 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 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:
- <uses-library android:name="com.google.android.maps" />
然后就是一些位置服务所需的权限:
- <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文件,代码如下:
- 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关键地方进行改写:
- GeoPoint point = LocationUtil.getFromLocationName(keyword);
大功告成,当我们搜索后把结果中的第一个显示到地图中,然后点击图标时,弹出提示。我们来看一下效果:
以上是关于android.location包基于位置服务中Geocoder的基本介绍,更多的内容会在以后找机会和大家再分享。
————————————————————————————————————————————————————————————————————
上次介绍了位置服务中的Geocoder,这次就来介绍一下LocationManager。LocationManager系统服务是位置服务的核心组件,它提供了一系列方法来处理与位置相关的问题,包括查询上一个已知位置、注册和注销来自某个LocationProvider的周期性的位置更新、注册和注销接近某个坐标时对一个已定义的Intent的触发等。今天我们就一起探讨一下LocationManager的简单应用。
在进入正题之前,朋友们需要了解与LocationManager相关的两个知识点:
provider:LocationManager获取位置信息的途径,常用的有两种:GPS和NETWORK。GPS定位更精确,缺点是只能在户外使用,耗电严重,并且返回用户位置信息的速度远不能满足用户需求。NETWORK通过基站和Wi-Fi信号来获取位置信息,室内室外均可用,速度更快,耗电更少。为了获取用户位置信息,我们可以使用其中一个,也可以同时使用两个。
LocationListener:位置监听器接口,定义了常见的provider状态变化和位置的变化的方法,我们需要实现此接口,完成自己的处理逻辑,然后让LocationManager注册此监听器,完成对各种状态的监听。
既然上面讲到位置服务的核心是LocationManager,那么我们如何取得一个LocationManager呢?像其他系统服务一样,通过以下方式即可得到一个LocationManager实例:
- LocationManager locMgr = (LocationManager) getSystemService(Context.LOCATION_SERVICE);
对象实例是获取到了,可是怎么应用呢?下面就通过一个示例具体演示一下。
我们新建一个location项目。因为示例是基于地图服务的,所以创建时别忘了Build Target要选中Google APIs这一项。
然后修改/res/layout/main.xml,代码如下:
- 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"/>
- <Button
- android:id="@+id/removeUpdates"
- android:layout_width="fill_parent"
- android:layout_height="wrap_content"
- android:text="removeUpdates"/>
- FrameLayout>
然后我们来看以下MainActivity.java文件,代码如下:
- package com.scott.location;
-
- import android.content.Context;
- import android.location.Location;
- import android.location.LocationListener;
- import android.location.LocationManager;
- import android.os.Bundle;
- import android.view.View;
- import android.widget.Button;
- 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;
-
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.main);
-
- mapView = (MapView) findViewById(R.id.mapView);
- mapView.getController().setZoom(17);
-
- final LocationManager locMgr = (LocationManager) getSystemService(Context.LOCATION_SERVICE);
-
-
- Location location = locMgr.getLastKnownLocation(LocationManager.GPS_PROVIDER);
- if (location != null) {
- markCurrLocation(location);
- }
-
- final MyLocationListener listener = new MyLocationListener();
-
- locMgr.requestLocationUpdates(LocationManager.GPS_PROVIDER, 5000, 5, listener);
-
- Button removeUpdates = (Button) findViewById(R.id.removeUpdates);
- removeUpdates.setOnClickListener(new View.OnClickListener() {
- @Override
- public void onClick(View v) {
-
- locMgr.removeUpdates(listener);
- }
- });
- }
-
-
-
-
-
- private void markCurrLocation(Location location) {
- mapView.removeAllViews();
- GeoPoint point = new GeoPoint((int) (location.getLatitude() * 1E6), (int) (location.getLongitude() * 1E6));
- 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, location manager!", Toast.LENGTH_SHORT).show();
- }
- });
- mapView.addView(marker, params);
- }
-
- @Override
- protected boolean isRouteDisplayed() {
- return false;
- }
-
- private final class MyLocationListener implements LocationListener {
-
- @Override
- public void onLocationChanged(Location location) {
- markCurrLocation(location);
- }
-
- @Override
- public void onStatusChanged(String provider, int status, Bundle extras) {
-
- }
-
- @Override
- public void onProviderEnabled(String provider) {
-
- }
-
- @Override
- public void onProviderDisabled(String provider) {
-
- }
-
- }
- }
因为用到了地图服务,所以需要在AndroidManifest.xml中的application标签之间加入google map library声明:
- <uses-library android:name="com.google.android.maps" />
然后加入位置服务所需的权限:
- <uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
这里朋友们需要注意:如果使用GPS_PROVIDER或者同时使用GPS_PROVIDER和NETWORK_PROVIDER,则只需声明ACCESS_FINE_LOCATION权限,它对于上述两个provider都是有效的;而ACCESS_COARSE_LOCATION权限只针对NETWORK_PROVIDER。
如果是在模拟器里调试的话,我们可以用以下两种方法设置一个模拟的坐标值:geo命令和DDMS。
先来说一下geo命令,它需要telnet到本机的5554端口,然后在命令行下输入geo fix命令,参数可附带经度、纬度和海拔(可选)。
具体操作如图:
如果朋友用的系统是windows7的话,会遇到一些小小的麻烦,因为windows7默认是没有装Telnet服务,所以我们需要手动安装一下,点击“开始->控制面板->程序->程序和功能”,然后再弹出的窗口左侧点击“打开或关闭Windows功能”,会弹出一下界面,选中“Telnet客户端”和“Telnet服务端”即可。如图:
不过,使用geo命令还是挺麻烦的,ADT提供了一个设置模拟坐标的界面,打开“Emulator Control”视图,即可看到一下界面:
如果设置了模拟坐标后,在模拟器的状态栏就会出现一个雷达图形的标志,如图: