提到Android基於位置的服務,就不得不提android.location包,location包提供了很便捷的API來實現基於位置的服務。主要包括Geocoder和LocationManager。今天就先來介紹一下Geocoder。
Geocoder可以在街道地址和經緯度地圖座標之間進行轉換。它提供了對兩種地理編碼功能的訪問:
Forward Geocoding(前向地理編碼):查詢某個地址的經緯度
Reverse Geocoding(反向地理編碼):查詢一個給定的經緯度所對應的街道地址。
分別對應以下方法:
1 2 |
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,程式碼如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 |
<?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檔案,程式碼如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 |
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:
1 |
<uses-library android:name="com.google.android.maps" /> |
然後就是一些位置服務所需的許可權:
1 2 3 |
<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檔案,程式碼如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 |
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關鍵地方進行改寫:
1 |
GeoPoint point = LocationUtil.getFromLocationName(keyword); |
大功告成,當我們搜尋後把結果中的第一個顯示到地圖中,然後點選圖示時,彈出提示。我們來看一下效果:
以上是關於android.location包基於位置服務中Geocoder的基本介紹,更多的內容會在以後找機會和大家再分享。