在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库:
在上面我标红的千万不要忘记。
layout下的main.xml:
下面是核心代码,重要的地方我做了注释:
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 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键,切换不同的模式:
上面是切换到了卫星模式。由于地图需要耗费大量的网络资源,如果网络比较慢的话会等待很长时间。