android定位和地圖開發實例
在android開發中地圖和定位是很多軟件不可或缺的內容,這些特色功能也給人們帶來了很多方便。
首先介紹一下地圖包中的主要類:
MapController : 主要控制地圖移動,伸縮,以某個GPS坐標為中心,控制MapView中的view組件,管理Overlay,提供View的基本功能。使用多種地圖模式(地圖模式(某些城市可實時對交通狀況進行更新),衛星模式,街景模式)來查看Google Map。常用方法:animateTo(GeoPoint point) setCenter(GeoPoint point) setZoom(int zoomLevel) 等。
Mapview : 是用來顯示地圖的view, 它派生自android.view.ViewGroup。當MapView獲得焦點,可以控制地圖的移動和縮放。地圖可以以不同的形式來顯示出來,如街景模式,衛星模式等,通過setSatellite(boolean) setTraffic(boolean), setStreetView(boolean) 方法。
Overlay : 是覆蓋到MapView的最上層,可以擴展其ondraw接口,自定義在MapView中顯示一些自己的東西。MapView通過MapView.getOverlays()對Overlay進行管理。
Projection :MapView中GPS坐標與設備坐標的轉換(GeoPoint和Point)。
定位系統包中的主要類:
LocationManager:本類提供訪問定位服務的功能,也提供獲取最佳定位提供者的功能。另外,臨近警報功能也可以借助該類來實現。
LocationProvider:該類是定位提供者的抽象類。定位提供者具備周期性報告設備地理位置的功能。
LocationListener:提供定位信息發生改變時的回調功能。必須事先在定位管理器中注冊監聽器對象。
Criteria:該類使得應用能夠通過在LocationProvider中設置的屬性來選擇合適的定位提供者。
Geocoder:用于處理地理編碼和反向地理編碼的類。地理編碼是指將地址或其他描述轉變為經度和緯度,反向地理編碼則是將經度和緯度轉變為地址或描述語言,其中包含了兩個構造函數,需要傳入經度和緯度的坐標。getFromLocation方法可以得到一組關于地址的數組。
下面開始地圖定位實例的開發,在開發地圖前需要 獲取Android地圖 API密鑰 網上有很多資料,這里就不再復述。
首先要在manifest.xml中設置全相應的權限和maps庫:
<application android:icon="@drawable/ic_launcher" android:label="@string/app_name" > <activity android:label="@string/app_name" android:name=".MyMapActivity" > <intent-filter > <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> <span style="color:#FF6666;"> <uses-library android:name="com.google.android.maps" /></span> </application> <span style="color:#FF6666;"> <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" /></span>在上面我標紅的千萬不要忘記。
layout下的main.xml:
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="fill_parent" android:orientation="vertical" > <com.google.android.maps.MapView android:id="@+id/mapview" android:layout_width="fill_parent" android:layout_height="fill_parent" android:apiKey="008uu0x2a7GWlK2LzCW872afBAPLhJ-U2R26Wgw" /> </LinearLayout>下面是核心代碼,重要的地方我做了注釋:
public class MyMapActivity extends MapActivity { /** Called when the activity is first created. */ private MapController mapController; private MapView mapView; private MyOverLay myOverLay; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); LocationManager locationManager=(LocationManager) getSystemService(Context.LOCATION_SERVICE); mapView=(MapView) this.findViewById(R.id.mapview); //設置交通模式 mapView.setTraffic(true); //設置衛星模式 mapView.setSatellite(false); //設置街景模式 mapView.setStreetView(false); //設置縮放控制 mapView.setBuiltInZoomControls(true); mapView.setClickable(true); mapView.setEnabled(true); //得到MapController實例 mapController=mapView.getController(); mapController.setZoom(15); myOverLay=new MyOverLay(); List<Overlay> overLays=mapView.getOverlays(); overLays.add(myOverLay); Criteria criteria=new Criteria(); criteria.setAccuracy(Criteria.ACCURACY_FINE); criteria.setAltitudeRequired(false); criteria.setBearingRequired(false); criteria.setCostAllowed(false); criteria.setPowerRequirement(Criteria.POWER_LOW); //取得效果最好的Criteria String provider=locationManager.getBestProvider(criteria, true); //得到Location Location location=locationManager.getLastKnownLocation(provider); updateWithLocation(location); //注冊一個周期性的更新,3秒一次 locationManager.requestLocationUpdates(provider, 3000, 0, locationListener); } @Override public boolean onCreateOptionsMenu(Menu menu) { // TODO Auto-generated method stub menu.add(0, 1, 1, "交通模式"); menu.add(0,2,2,"衛星模式"); menu.add(0,3,3,"街景模式"); return super.onCreateOptionsMenu(menu); } @Override public boolean onOptionsItemSelected(MenuItem item) { // TODO Auto-generated method stub super.onOptionsItemSelected(item); switch (item.getItemId()) { case 1://交通模式 mapView.setTraffic(true); mapView.setSatellite(false); mapView.setStreetView(false); break; case 2://衛星模式 mapView.setSatellite(true); mapView.setStreetView(false); mapView.setTraffic(false); break; case 3://街景模式 mapView.setStreetView(true); mapView.setTraffic(false); mapView.setSatellite(false); break; default: mapView.setTraffic(true); mapView.setSatellite(false); mapView.setStreetView(false); break; } return true; } private void updateWithLocation(Location location){ if(location!=null){ //為繪制類設置坐標 myOverLay.setLocation(location); GeoPoint geoPoint=new GeoPoint((int)(location.getLatitude()*1E6), (int)(location.getLongitude()*1E6)); //定位到指定的坐標 mapController.animateTo(geoPoint); mapController.setZoom(15); } } private final LocationListener locationListener=new LocationListener() { @Override public void onStatusChanged(String provider, int status, Bundle extras) { // TODO Auto-generated method stub } @Override public void onProviderEnabled(String provider) { // TODO Auto-generated method stub } @Override public void onProviderDisabled(String provider) { // TODO Auto-generated method stub } //當坐標改變時出發此函數 @Override public void onLocationChanged(Location location) { // TODO Auto-generated method stub updateWithLocation(location); } }; class MyOverLay extends Overlay{ private Location location; public void setLocation(Location location){ this.location=location; } @Override public boolean draw(Canvas canvas, MapView mapView, boolean shadow, long when) { // TODO Auto-generated method stub super.draw(canvas, mapView, shadow); Paint paint=new Paint(); Point myScreen=new Point(); //將經緯度換成實際屏幕的坐標。 GeoPoint geoPoint=new GeoPoint((int)(location.getLatitude()*1E6), (int)(location.getLongitude()*1E6)); mapView.getProjection().toPixels(geoPoint, myScreen); paint.setStrokeWidth(1); paint.setARGB(255, 255, 0, 0); paint.setStyle(Paint.Style.STROKE); Bitmap bmp=BitmapFactory.decodeResource(getResources(), R.drawable.mypicture); //把這張圖片畫到相應的位置。 canvas.drawBitmap(bmp, myScreen.x, myScreen.y,paint); canvas.drawText("天堂沒有路", myScreen.x, myScreen.y, paint); return true; } } @Override protected boolean isRouteDisplayed() { // TODO Auto-generated method stub return false; } @Override public boolean onKeyDown(int keyCode, KeyEvent event) { // TODO Auto-generated method stub if (keyCode == KeyEvent.KEYCODE_BACK) { AlertDialog.Builder builder = new AlertDialog.Builder(this); builder.setMessage("你確定退出嗎?") .setCancelable(false) .setPositiveButton("確定", new DialogInterface.OnClickListener() { public void onClick(DialogInterface dialog, int id) { MyMapActivity.this.finish(); android.os.Process .killProcess(android.os.Process .myPid()); android.os.Process.killProcess(android.os.Process.myTid()); android.os.Process.killProcess(android.os.Process.myUid()); } }) .setNegativeButton("返回", new DialogInterface.OnClickListener() { public void onClick(DialogInterface dialog, int id) { dialog.cancel(); } }); AlertDialog alert = builder.create(); alert.show(); return true; } return super.onKeyDown(keyCode, event); } }接下來看一下運行后效果:
可以放大縮小:
可是使用menu鍵,切換不同的模式:
轉自:http://blog.csdn.net/wangkuifeng0118/article/details/7244364