iOS原生地图开发详解
在上一篇博客中:http://my.oschina.net/u/2340880/blog/414760。对iOS中的定位服务进行了详细的介绍与参数说明,在开发中,地位服务往往与地图框架结合使用,这篇博客主要对iOS官方的地图框架MapKit.framework进行介绍。
一、初始化地图视图与相关属性方法介绍
1、初始化地图视图
地图视图的展示依赖于MKMapView这个类,这个类继承于UIView,因此和其他View的使用方法类似。在我们需要展现地图的地方:
- (void)viewDidLoad { [super viewDidLoad]; MKMapView * mapView =[[MKMapView alloc]initWithFrame:self.view.frame]; [self.view addSubview:mapView];}
运行发现,一张世界地图就在我们的设备上了,apple内置的地图数据是由高德提供的。
2、系统提供的三种地图样式
可以通过MKMapView的mapType这个属性设置地图的模式:
@property (nonatomic) MKMapType mapType;
枚举如下:
typedef NS_ENUM(NSUInteger, MKMapType) { MKMapTypeStandard = 0,//标准式的行政地图(会显示城市,街道等) MKMapTypeSatellite,//标准的卫星地图 MKMapTypeHybrid//混合地图(在卫星图上显示街道等名称)};
3、设置地图的中心和比例尺
在百度地图等第三方地图服务的SDK中,都会提供一个类似zoomLevel比例尺的属性。通过官方的API设置这个属性有些麻烦,但是也更加灵活。首先,设置地图的中心位置和比例尺是通过region这个属性实现的。region结构体如下:
typedef struct { CLLocationCoordinate2D center;//地图中心的经纬度 MKCoordinateSpan span;//地图显示的经纬度范围} MKCoordinateRegion;
这个结构体中包含了两个结构体,其中CLLocationCoordinate2D很好理解,就是简单的经纬度,解释如下:
typedef struct { CLLocationDegrees latitude;//纬度,北纬为正,南纬为负 CLLocationDegrees longitude;//经度,东经为正,西经为负} CLLocationCoordinate2D;
MKCoordinateSpan这个结构体比较复杂,如下:
typedef struct { CLLocationDegrees latitudeDelta;//纬度范围 CLLocationDegrees longitudeDelta;//经度范围} MKCoordinateSpan;
这个结构体定义的应该是一个范围,因为北纬南纬加起来180°,所以纬度范围的取值应为0-180。同理,经度范围的取值范围为0-360。
通过上面的介绍,我们举个例子,将北京市设为地图的中心区域,并且比例设置为显示北京大小。通过百度,首先知道北京市界的地理坐标为:北纬39”26’至41”03’,东经115”25’至 117”30’。北京市区坐标为:北纬39.9”,东经116. 3”。代码如下:
mapView.region=MKCoordinateRegionMake(CLLocationCoordinate2DMake(39.26, 116.3), MKCoordinateSpanMake(1.8, 2.05));
运行后可以看到,北京市基本上是在地图中心的,效果如下:

注意:MKCoordinateSpan的显示范围是取决于大的一边的,比如如果我们这样写:
MKCoordinateSpanMake(1.8, 360);
最后依然会显示整个世界地图。
- (void)setRegion:(MKCoordinateRegion)region animated:(BOOL)animated;
这个方法可以在设置后给地图加上动画效果
@property (nonatomic) CLLocationCoordinate2D centerCoordinate;
设置地图的中心点位置
- (void)setCenterCoordinate:(CLLocationCoordinate2D)coordinate animated:(BOOL)animated;
设置地图的中心点位置,并附带动画效果
4、坐标转换方法
- (CGPoint)convertCoordinate:(CLLocationCoordinate2D)coordinate toPointToView:(UIView *)view;
将经纬度转换为视图上的坐标
- (CLLocationCoordinate2D)convertPoint:(CGPoint)point toCoordinateFromView:(UIView *)view;
将视图上的坐标转换为经纬度
- (CGRect)convertRegion:(MKCoordinateRegion)region toRectToView:(UIView *)view;
将地理显示的区域转换为视图上的坐标区域
- (MKCoordinateRegion)convertRect:(CGRect)rect toRegionFromView:(UIView *)view;
将视图上的坐标区域转换为地理区域
5、MKMapView常用方法和属性
@property (nonatomic, getter=isZoomEnabled) BOOL zoomEnabled;
设置是否允许捏合手势进行地图缩放
@property (nonatomic, getter=isScrollEnabled) BOOL scrollEnabled;
设置是否允许滑动
@property (nonatomic, getter=isRotateEnabled) BOOL rotateEnabled;
设置是否允许旋转地图
@property (nonatomic, getter=isPitchEnabled) BOOL pitchEnabled;
设置是否支持3D效果
@property (nonatomic) BOOL showsPointsOfInterest;
设置是否显示兴趣点,例如学校,医院等
@property (nonatomic) BOOL showsBuildings;
设置是否显示建筑物轮廓,只在标准的地图中有效
@property (nonatomic) BOOL showsUserLocation;
是否显示用户位置
@property (nonatomic) MKUserTrackingMode userTrackingMode;
- (void)setUserTrackingMode:(MKUserTrackingMode)mode animated:(BOOL)animated;
设置更新用户位置的模式,当显示用户位置设置为YES,这个方法也设置了后,地图框架为我们直接集成了定位,地图上就会显示我们的位置,模式的枚举如下:
typedef NS_ENUM(NSInteger, MKUserTrackingMode) { MKUserTrackingModeNone = 0, // 不跟踪用户位置 MKUserTrackingModeFollow, // 跟踪用户位置 MKUserTrackingModeFollowWithHeading, // 当方向改变时跟踪用户位置}
@property (nonatomic, readonly) MKUserLocation *userLocation;
获取用户位置的标注
@property (nonatomic, readonly, getter=isUserLocationVisible) BOOL userLocationVisible;
获取用户位置是否可见
- (void)addAnnotation:(id <MKAnnotation>)annotation;
在地图上添加一个标注
- (void)addAnnotations:(NSArray *)annotations;
在地图上添加一组标注
- (void)removeAnnotation:(id <MKAnnotation>)annotation;
移除一个标注
- (void)removeAnnotations:(NSArray *)annotations;
移除一组标注
@property (nonatomic, readonly) NSArray *annotations;
获取所有标注数组
- (MKAnnotationView *)viewForAnnotation:(id <MKAnnotation>)annotation;
获取标注的视图
- (MKAnnotationView *)dequeueReusableAnnotationViewWithIdentifier:(NSString *)identifier;
获取复用的标注
- (void)selectAnnotation:(id <MKAnnotation>)annotation animated:(BOOL)animated;
选中一个标注
- (void)deselectAnnotation:(id <MKAnnotation>)annotation animated:(BOOL)animated;
取消选中一个标注
@property (nonatomic, copy) NSArray *selectedAnnotations;
选中标注的数组
- (void)addOverlay:(id <MKOverlay>)overlay level:(MKOverlayLevel)level;
添加一个地图覆盖物,level是设置一个层级,枚举如下:
typedef NS_ENUM(NSInteger, MKOverlayLevel) { MKOverlayLevelAboveRoads = 0, // 覆盖物位于道路之上 MKOverlayLevelAboveLabels//覆盖物位于标签之上}
- (void)addOverlays:(NSArray *)overlays level:(MKOverlayLevel)level;
添加一组地图覆盖物
- (void)removeOverlay:(id <MKOverlay>)overlay;
移除一个地图覆盖物
- (void)removeOverlays:(NSArray *)overlays;
移除一组地图覆盖物
- (void)insertOverlay:(id <MKOverlay>)overlay atIndex:(NSUInteger)index level:(MKOverlayLevel)level;
在索引处插入一个地图覆盖物
- (void)insertOverlay:(id <MKOverlay>)overlay aboveOverlay:(id <MKOverlay>)sibling;
将一个地图覆盖物插在到某个覆盖物之上
- (void)insertOverlay:(id <MKOverlay>)overlay belowOverlay:(id <MKOverlay>)sibling;
将一个地图覆盖物插入到某个覆盖物之下
- (void)exchangeOverlay:(id <MKOverlay>)overlay1 withOverlay:(id <MKOverlay>)overlay2;
替换一个地图覆盖物
@property (nonatomic, readonly) NSArray *overlays;
地图覆盖物数组
- (NSArray *)overlaysInLevel:(MKOverlayLevel)level;
层级属性下的东土覆盖物数组
二、MKMapViewDelegate相关方法解读
- (void)mapView:(MKMapView *)mapView regionWillChangeAnimated:(BOOL)animated;
地图显示位置将要改变时调用的方法
- (void)mapView:(MKMapView *)mapView regionDidChangeAnimated:(BOOL)animated;
地图显示位置已经改变时调用的方法
- (void)mapViewWillStartLoadingMap:(MKMapView *)mapView;
地图将要加载时调用的方法
- (void)mapViewDidFinishLoadingMap:(MKMapView *)mapView;
地图加载完成时执行的方法
- (void)mapViewDidFailLoadingMap:(MKMapView *)mapView withError:(NSError *)error;
地图加载失败时执行的方法
- (MKAnnotationView *)mapView:(MKMapView *)mapView viewForAnnotation:(id <MKAnnotation>)annotation;
渲染标注视图时调用的方法,可以通过这个方法自定义标注视图
- (void)mapView:(MKMapView *)mapView didAddAnnotationViews:(NSArray *)views;
标注添加完成后调用的方法
- (void)mapView:(MKMapView *)mapView didSelectAnnotationView:(MKAnnotationView *)view;
选中标注时调用的方法
- (void)mapView:(MKMapView *)mapView didDeselectAnnotationView:(MKAnnotationView *)view;
取消选中标注时调用的方法
- (void)mapViewWillStartLocatingUser:(MKMapView *)mapView;
将要开始定位用户位置时调用的方法
- (void)mapViewDidStopLocatingUser:(MKMapView *)mapView;
停止定位用户位置时调用的方法
- (void)mapView:(MKMapView *)mapView didUpdateUserLocation:(MKUserLocation *)userLocation;
更新用户位置时调用的方法
- (void)mapView:(MKMapView *)mapView didFailToLocateUserWithError:(NSError *)error;
更新用户位置失败时调用的方法
- (void)mapView:(MKMapView *)mapView annotationView:(MKAnnotationView *)view didChangeDragState:(MKAnnotationViewDragState)newState
fromOldState:(MKAnnotationViewDragState)oldState;
标注拖动状态改变调用的方法,MKAnnotationViewDragState的枚举如下:
typedef NS_ENUM(NSUInteger, MKAnnotationViewDragState) { MKAnnotationViewDragStateNone = 0, // 初始状态 MKAnnotationViewDragStateStarting, // 开始拖动时 MKAnnotationViewDragStateDragging, // 正在拖动 MKAnnotationViewDragStateCanceling, // 取消拖动 MKAnnotationViewDragStateEnding // 结束拖动};
- (void)mapView:(MKMapView *)mapView didChangeUserTrackingMode:(MKUserTrackingMode)mode animated:(BOOL)animated;
定位用户位置模式改变时调用的方法
- (MKOverlayView *)mapView:(MKMapView *)mapView viewForOverlay:(id <MKOverlay>)overlay;
渲染覆盖物视图时调用的方法,可以自定义覆盖物视图
- (void)mapView:(MKMapView *)mapView didAddOverlayViews:(NSArray *)overlayViews;
添加完成覆盖物数组执行的方法
原文链接:iOS原生地图开发详解
iOS原生地图开发详解的更多相关文章
- iOS原生地图开发进阶——使用导航和附近兴趣点检索
iOS原生地图开发进阶——使用导航和附近兴趣点检索 iOS中的mapKit框架对国际化的支持非常出色.在前些篇博客中,对这个地图框架的基础用法和标注与覆盖物的添加进行了详细的介绍,这篇博客将介绍两个更 ...
- iOS原生地图开发指南续——大头针与自定义标注
iOS原生地图开发指南续——大头针与自定义标注 出自:http://www.sxt.cn/info-6042-u-7372.html 在上一篇博客中http://my.oschina.net/u/23 ...
- iOS 原生地图 开发
iOS开发有时候用到地图,不少人第一想到的是用第三方.当然有时候为了和安卓同步,可能会一起使用某一第三方.但有时候,我们是可以用原生地图开发的.上面两个示意图是原生地图的自定义开发.运行demo,将展 ...
- iOS 百度地图使用详解
最近仿照美团做了款应用,刚好用到百度地图,高德地图之前用的比较多,只是这个项目的后台服务器是另外一个公司做的,他们用的就是百度地图,现在网上用百度地图的还不算太多,博文也是断断续续的,主要是中间跳跃有 ...
- iOS索引列开发详解
做苹果开发的朋友在地区列表可能会遇到在页面的右侧有一列类似与导航的索引列,这次有机会遇到了,细细研究了一下,原来没有想象中的困难,只需要简单的几步就能做出自己的索引列.本来想和搜索条在一块讲解,后来考 ...
- iOS应用开发详解
<iOS应用开发详解> 基本信息 作者: 郭宏志 出版社:电子工业出版社 ISBN:9787121207075 上架时间:2013-6-28 出版日期:2013 年7月 开本:16开 ...
- [转载]Apple Watch 开发详解
Apple Watch 开发详解 Apple Watch现在对于第三方开发者来说更多的还是一块额外的屏幕.暂时WatchKit没有能给出足够的接口.现在Watch App的主要运算逻辑需要依赖iPho ...
- Apple Watch 开发详解
Apple Watch 开发详解 Apple Watch现在对于第三方开发者来说更多的还是一块额外的屏幕.暂时WatchKit没有能给出足够的接口.现在Watch App的主要运算逻辑需要依赖iPho ...
- javacv开发详解之1:调用本机摄像头视频(建议使用javaCV最新版本)
javaCV系列文章: javacv开发详解之1:调用本机摄像头视频 javaCV开发详解之2:推流器实现,推本地摄像头视频到流媒体服务器以及摄像头录制视频功能实现(基于javaCV-FFMPEG.j ...
随机推荐
- 对抗静态分析——so文件的加密
[预备起~~~]最近在忙找工作的事情,笔试~面试~笔试~面试...很久没有写(pian)文(gao)章(fei).忙了一阵子之后,终于~~~到了选offer的阶段(你家公司不是牛吗,老子不接你家off ...
- FDO error:Failed to label layer(XXX) for class Default
描述: A column was specified that does not exist. 出现这个问题的原因在于label features 展示的字段不存在或者为空,只要将其勾选去掉或者换个显 ...
- Unable to execute dex: Multiple dex files define Lcom/kenai/jbosh/AbstractAttr
出现该问题应该是导入项目的android版本问题. 编译的时候把build path 下 source选项卡中的libs去掉就正常了. http://blog.csdn.net/e421083 ...
- [leetcode] Bitwise AND of Numbers Range
Bitwise AND of Numbers Range Given a range [m, n] where 0 <= m <= n <= 2147483647, return t ...
- Android—SQLITE数据库的设计和升降级
Google为Andriod的较大的数据处理提供了SQLite,他在数据存储.管理.维护等各方面都相当出色,功能也非常的强大.SQLite具备下列特点: 1.轻量级 使用 SQLite 只需要带一个动 ...
- 用thinkPHP实现验证码的功能
许多系统的登录都有验证码,而如果使用thinkPHP框架搭建网站的话,验证码的生成和验证就比较容易了 1.生成验证码 thinkPHP有对应生成验证码的方法 要使用验证码,需要导入扩展类库中的ORG. ...
- git 查看远程分支、本地分支、删除本地分支
1 查看远程分支 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 $ git branch -a * br-2.1.2.2 master remotes/origi ...
- mysql中FIND_IN_SET的使用方法
在mysql中,有时我们在做数据库查询时,需要得到某字段中包含某个值的记录,但是它也不是用like能解决的,使用like可能查到我们不想要的记录,它比like更精准,这时候mysql的FIND_IN_ ...
- 管理故事——和尚挑水的故事
有时候企业.公司的各种混乱都是源于管理问题,例如人浮于事.资源错配.机构臃肿-----,暂且不说企业管理.项目的管理,光是个人工作的管理.一个处理不好,接踵而来的就是一堆问题,可怕的不是出现问题,而是 ...
- 使用cxf构建webservice
一.简介 cxf是apache下的一个开源的,功能齐全的WebService框架.是由Celtix和XFire两个开源框架组合而成.cxf可以帮助我们使用前端编程模型构建和开发服务,如JAX- WS和 ...