android定位和地图开发实例

fmms 13年前

             在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进行管理。

          ProjectionMapViewGPS坐标与设备坐标的转换(GeoPointPoint)。

          定位系统包中的主要类:

         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);   }  }
         接下来看一下运行后效果:

       android定位和地图开发实例

     可以放大缩小:

   android定位和地图开发实例

    可是使用menu键,切换不同的模式:

    android定位和地图开发实例

   上面是切换到了卫星模式。由于地图需要耗费大量的网络资源,如果网络比较慢的话会等待很长时间。
转自:http://blog.csdn.net/wangkuifeng0118/article/details/7244364