iOS开发--使用地图规划路径

iOS系统为我们提供了原生的mapView,我们只需要将mapView控件拖动到stiryBoard中,或者用代码创建并添加到界面上,就能将地图显示出来。然后再通过实现地图的代理方法,来实现路线规划,用户位置定位,跟随用户位置等功能。下面展示的代码就是关于如何在界面上添加一个地图,并通过地图的代理方法来实现路线规划。
实现原理是我们先创建一个编码工具类,这个工具类的作用是可以用来创建起点和终点。当我们创建好起点终点后,就通过一个MKDirections类的对象将我们的起点和终点发送给服务器,苹果的服务器收到我们提供的起点和终点后会帮我们规划好一条路线,并返回给我们。这个过程中app并不参与路线规划的计算,app只是将用户输入的起点终点发送给服务器,然后由服务器来进行路线规划。这样最直接的好处是能够将路径计算的任务交给服务器,从而提高了app的性能。当然除了苹果提供的原生地图,我们也可以使用百度地图和高德地图的SDK。但要注意的是苹果地图在国内的数据供应商是高德。所以使用高德地图SDK和使用苹果原生地图数据是一样的。使用第三方地图的SDK还会产生一个问题就是会降低app的性能。运行app进行定位和路线规划的时候内存会飙升到百兆以上。但这并不是说我们不能使用第三方地图的SDK,只是具体使用那一个,需要根据我们开发中的需求来确定。第三方地图的优势在于有更丰富的功能。像百度地图和高德地图就提供了室内地图,热力图等功能。


#import "ViewController.h"
//地图的方法,需要导入地图的头文件

#import "HMAnnotation.h"

@interface ViewController ()
//编码工具
@property(strong,nonatomic)CLGeocoder *geocoder;
//用于发送请求给服务器,获取规划好后的路线。
@property(strong,nonatomic)MKDirections *directs;
//地图
@property(strong,nonatomic)MKMapView *mapView;

@end
- (void)viewDidLoad {
 [super viewDidLoad];
 
// 创建并添加地图
 MKMapView *mapView = [[MKMapView alloc]initWithFrame:self.view.bounds];
 self.mapView = mapView;
 [self.view addSubview:mapView];
 mapView.delegate = self;
 
 self.geocoder = [[CLGeocoder alloc]init];
// 通过地理编码的类,创建起点和终点
 [_geocoder geocodeAddressString:@"华强北" completionHandler:^(NSArray * _Nullable placemarks, NSError * _Nullable error) {
  
  MKMapItem *item1 = [[MKMapItem alloc]initWithPlacemark:[[MKPlacemark alloc]initWithPlacemark:placemarks.lastObject]];
  
//  添加一个大头针
//  创建一个大头针模型
  HMAnnotation *anni = [[HMAnnotation alloc]init];
//  设置大头针的coordinate属性(即经纬度属性),coordinate属性类型为经纬度类型.
  anni.coordinate = item1.placemark.location.coordinate;
//  将大头针添加到地图
  [self.mapView addAnnotation:anni];
  
//  让地图显示到定位的点的附近,region表示一个范围
  MKCoordinateRegion region = MKCoordinateRegionMake(item1.placemark.coordinate, MKCoordinateSpanMake(0.01, 0.01));
//  让地图只显示一个固定要显示的区域
  self.mapView.region = region;
//     创建终点  
  [_geocoder geocodeAddressString:@"东莞" completionHandler:^(NSArray * _Nullable placemarks, NSError * _Nullable error) {
 //    item可以理解为点的意思,这里为了可读性,加尾缀1和2来表示 
   MKMapItem *item2 = [[MKMapItem alloc]initWithPlacemark:[[MKPlacemark alloc]initWithPlacemark:placemarks.lastObject]];
   
//   添加一个大头针到终点
   HMAnnotation *anni = [[HMAnnotation alloc]init];
   anni.coordinate = item1.placemark.location.coordinate;
   [self.mapView addAnnotation:anni];
   
//   调用下面自定义方法给服务器发送请求
   [self requesrWithItem:item1 andItem2:item2];
  }];
 }];
}

-(void)requesrWithItem:(MKMapItem *)item1 andItem2:(MKMapItem *)item2
{

//  接收传入的参数,向苹果服务器发送请求
//  创建请求体,设置请求体的总店和起点
 MKDirectionsRequest *requst = [[MKDirectionsRequest alloc]init];
 requst.source = item1;
 requst.destination = item2;
 
// 发送请求
 self.directs = [[MKDirections alloc]initWithRequest:requst];
// 服务器计算好路径后给我们返回了数据
 [self.directs calculateDirectionsWithCompletionHandler:^(MKDirectionsResponse * _Nullable response, NSError * _Nullable error) {
  
//  获取所有规划的路径
  NSArray *routes = response.routes;
//  获取规划路径中的的最后一条路径,如果只返回一条路径就能取到那一条
  MKRoute *route = routes.lastObject;
//  定义数组保存路线中的每一步
  NSArray *stepArr = route.steps;
//  遍历路线中的每一步
  for (MKRouteStep *step in stepArr) {
//   打印路线的信息
   NSLog(@"%f %@",step.distance,step.instructions);
//   绘制遮盖打印到地图上,绘制方法的实现在下面
   [self.mapView addOverlay:step.polyline];
  }
 }];
}
// 返回指定的遮盖模型所对应的遮盖视图, renderer-渲染
- (MKOverlayRenderer *)mapView:(MKMapView *)mapView rendererForOverlay:(id)overlay
{
// 判断类型是不是MKOverlay类型
 if ([overlay isKindOfClass:[MKPolyline class]]) {
//  针对线段,系统有提供好的遮盖样式
  MKPolylineRenderer *render = [[MKPolylineRenderer alloc]initWithOverlay:overlay];
//  配置渲染的宽度和渲染的颜色
  render.lineWidth = 5;
  render.strokeColor = [UIColor redColor];
//  返回配置好的渲染
  return render;
 }
//  否则返回nil
  return nil;
}

这篇文章针对的是掌握iOS中掌握UI基本使用方法,还有能理解OC面向对象的概念的开发者。如果你还没掌握上面的内容,那看不懂代码和文章内容是正常的现象。如果能看懂但对于文章里面的一个类还不能很好地理解,可以去查看苹果的官方文档。在官方文档里对于各个类的介绍和使用都有详细的说明。如果想看代码的运行效果,可以将代码拷贝到一个新建项目的ViewController.m文件中,运行后就能看到效果。
这个demo已经上传到git上面了。下面是链接地址,点击链接地址再点击downLoadZIP就可以下载整个文件。直接运行就可以看到效果了。
项目地址

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,907评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,987评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,298评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,586评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,633评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,488评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,275评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,176评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,619评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,819评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,932评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,655评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,265评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,871评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,994评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,095评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,884评论 2 354

推荐阅读更多精彩内容