虽然是转载的,还是说几句吧。网上百度地图导航路线规划倒是挺多的,苹果自带的高德导航确实挺少,研究了好久发现就这个讲的稍微全一点,把需要用到的类什么的都讲清楚了。不过高德有个方法可以跳转到它自己的地图上,给个起点终点自动帮你计算路线,还有文字标注,这点倒是很方便。
[MKMapItem openMapsWithItems:items launchOptions:dictM];//这个方法就是跳转到自带地图的,不用手机上安装有高德地图也可以跳转。还是把具体代码贴出来,网上其它地方有,这里就当总结一下。
/**
* 开始导航
*/
- (void)startNavigation{
//1,获取用户输入的起点终点
//2,利用GEO对象进行地理编码获取地标对象
CLLocationCoordinate2D coords = CLLocationCoordinate2DMake([InforCenter shareInfoCenter].lantitude, [InforCenter shareInfoCenter].longitude);
CLLocation * currentLocation = [[CLLocation alloc]initWithLatitude:coords.latitude longitude:coords.longitude];
//2.1 获取开始位置的地标
[self.geocoder reverseGeocodeLocation:currentLocation completionHandler:^(NSArray<CLPlacemark *> * _Nullable placemarks, NSError * _Nullable error) {
if (placemarks.count ==0 ||error != nil) {
return ;
}
//开始位置的地标
CLPlacemark * startPlacemark = [placemarks firstObject];
//3,获得结束位置的地标
float latitude = [[NSString stringWithFormat:@"%@",self.businessDic[@"Y"]] floatValue];
float lOngitude= [[NSString stringWithFormat:@"%@",self.businessDic[@"X"]] floatValue];
CLLocationCoordinate2D coords1 = CLLocationCoordinate2DMake(latitude,longitude);
CLLocation * currentLocation1 = [[CLLocation alloc]initWithLatitude:coords1.latitude longitude:coords1.longitude];
[self.geocoder reverseGeocodeLocation:currentLocation1 completionHandler:^(NSArray<CLPlacemark *> * _Nullable placemarks, NSError * _Nullable error) {
if (placemarks.count ==0 ||error != nil) {
return ;
}
CLPlacemark * endPlacemark = [placemarks firstObject];
//4,获得地标后开始导航
[self startNavigationWithStartPlacemark:startPlacemark endPlacemark:endPlacemark];
}];
}];
}
/**
* 利用地标位置开始设置导航
*
* @param startPlacemark 开始位置的地标
* @param endPlacemark 结束位置的地标
*/
-(void)startNavigationWithStartPlacemark:(CLPlacemark *)startPlacemark endPlacemark:(CLPlacemark*)endPlacemark
{
//0,创建起点
MKPlacemark * startMKPlacemark = [[MKPlacemark alloc]initWithPlacemark:startPlacemark];
//0,创建终点
MKPlacemark * endMKPlacemark = [[MKPlacemark alloc]initWithPlacemark:endPlacemark];
//1,设置起点位置
MKMapItem * startItem = [[MKMapItem alloc]initWithPlacemark:startMKPlacemark];
//2,设置终点位置
MKMapItem * endItem = [[MKMapItem alloc]initWithPlacemark:endMKPlacemark];
//3,起点,终点数组
NSArray * items = @[ startItem ,endItem];
//4,设置地图的附加参数,是个字典
NSMutableDictionary * dictM = [NSMutableDictionary dictionary];
//导航模式(驾车,步行)
dictM[MKLaunchOptionsDirectionsModeKey] = MKLaunchOptionsDirectionsModeDriving;
//地图显示的模式
dictM[MKLaunchOptionsMapTypeKey] = MKMapTypeStandard;
//只要调用MKMapItem的open方法,就可以调用系统自带地图的导航
//Items:告诉系统地图从哪到哪
//launchOptions:启动地图APP参数(导航的模式/是否需要先交通状况/地图的模式/..)
[MKMapItem openMapsWithItems:items launchOptions:dictM];
}
//这里是自己计算路线,相当于把起点和终点用线连接,选择不同的模式就会有不同的路线。
- (IBAction)goSearch {
CLLocationCoordinate2D fromCoordinate = _coordinate;
CLLocationCoordinate2D toCoordinate = CLLocationCoordinate2DMake(32.010241,
118.719635);
MKPlacemark *fromPlacemark = [[MKPlacemark alloc] initWithCoordinate:fromCoordinate
addressDictionary:nil];
MKPlacemark *toPlacemark = [[MKPlacemark alloc] initWithCoordinate:toCoordinate
addressDictionary:nil];
MKMapItem *fromItem = [[MKMapItem alloc] initWithPlacemark:fromPlacemark];
MKMapItem *toItem = [[MKMapItem alloc] initWithPlacemark:toPlacemark];
[self findDirectionsFrom:fromItem
to:toItem];
}
#pragma mark - Private
- (void)findDirectionsFrom:(MKMapItem *)source
to:(MKMapItem *)destination
{
MKDirectionsRequest *request = [[MKDirectionsRequest alloc] init];
request.source = source;
request.destination = destination;
request.requestsAlternateRoutes = YES;
MKDirections *directiOns= [[MKDirections alloc] initWithRequest:request];
[directions calculateDirectionsWithCompletionHandler:
^(MKDirectionsResponse *response, NSError *error) {
if (error) {
NSLog(@"error:%@", error);
}
else {
MKRoute *route = response.routes[0];
[self.mapView addOverlay:route.polyline];
}
}];
}
}
介绍
IOS7在mapping框架中可以看到MapKit的一些改变和新增。其中一个关键的例子,新增一个API可以提供两点之间的路线指南。今天我们将建立一个简单的选择路线的应用来看一下如何使用这个API。我们也会简要的介绍一下叠加渲染的API。
请求方向指南
MapKit中我们需要许多不同的类,但通过依次使用它们会觉得非常简单的。为了查询苹果服务器方向指南集合,我们需要MKDirectionsRequest 对象的封装细节。这个类在IOS6中已经被应用使用了,能够来生成自己的行车方向指南。在IOS7已经进一步被扩展,可以允许开发者从苹果服务器中请求方向指南。
MKDirectionsRequest *directiOnsRequest= [MKDirectionsRequest new];
为了创建一个请求,我们需要设置源和目的地,两个地点都是MKMapItem 对象。这些对象都代表地图上的位置,包括它的地点以及一些元数据,比如名称,手机号码和URL。有几种方法可以用来创建这个对象,其中一种可以使用用户当前的地点;
MKMapItem *source = [MKMapItem mapItemForCurrentLocation];
如果用户第一次运行这个程序,他们将被询问是否允许使用他们当前位置:
你也可以用initWithPlacemark:方法使用自定义位置来创建一个地图item,这也引入了一个新的类。MKPlacemark代表地图上的实际位置,也就是它的经纬度。我们可以使用一个反向地理编码从CoreLocation来生成一个地标,但因为那不是本文的重点,所以我们将用一些固定的坐标创建地标。把所有的放在一起我们就可以完成一个MKDirectionsRequest 对象了。
// Make the destination CLLocationCoordinate2D destinatiOnCoords= CLLocationCoordinate2DMake(38.8977, -77.0365); MKPlacemark *destinatiOnPlacemark= [[MKPlacemark alloc] initWithCoordinate:destinationCoords addressDictionary:nil]; MKMapItem *destination = [[MKMapItem alloc] initWithPlacemark:destinationPlacemark]; // Set the source and destination on the request [directionsRequest setSource:source]; [directionsRequest setDestination:destination];
MKDirectionsRequest 有一些其他控制返回路线信息的属性,如下:
1、 departureDate 和arrivalDate。设置这些值,由于旅行时间的限制,将优化返回的路线,例如,会考虑到标准的路况信息。
2、 TransportType。目前苹果通过枚举值MKDirectionsTransportTypeAutomobile 或者MKDirectionsTransportTypeWalking提供步行或者驾车方式。默认值是MKDirectionsTransportTypeAny。
3、 RequestsAlternateRoutes。如果路由服务器可以找出多条合理的路线,设置YES将会返回所有路线。否则,只返回一条路线。
现在我们已经有了一个可用的请求,可以发送去请求路线。需要使用MKDirections 类,它有一个用MKDirectionsRequest对象构造的函数:
MKDirections *directiOns= [[MKDirections alloc] initWithRequest:directionsRequest];
将使用两个方法:calculateETAWithCompletionHandler:计算路线的花费的时间,calculateDirectionsWithCompletionHandler计算真实的路线。两个方法都是异步的,并有一个completion handling块。MKDirections 对象也有一个取消方法,提供给当前任何正在运行的请求。还有calculating属性,如果当前有个请求正在执行,返回为true。单个的MKDirections对象一次只能运行一个请求,额外去请求将会失败。如果你想运行多个并发的请求,你需要创建多个MKDirections对象。但注意,请求太多的话可能会导致苹果服务器因为节流返回错误。
[directions calculateDirectionsWithCompletionHandler:^(MKDirectionsResponse *response, NSError *error) { // Handle the response here }];
方向指南响应
从苹果服务器回送的对象是MKDirectionsResponse,还有源和目的地,包括MKRoute对象数组。注意,这个数组只有一个对象除非我们设置setrequestsAlternateRoutes为YES。
MKRoute 对象,如它的名字,代表用户选择的两点之间的路线。它包含一些关于路线信息的属性:
1、 name:从服务器找到路线时自动生成的。它是基于路线的重要特征。
2、 advisoryNoties:字符串数组,包含一些适合生成路线的警告等诸如此类的详情。
3、 distance:是沿着路线的距离,不是位移。单位是米。
4、 expectedTravelTime:NSTimeInterval,单位秒。
5、 transportType:
6、 polyline:MKPolyline代表地图上路径。可以画在MKMapView上,下一节将会看到。
7、 steps:MKRouteStep 对象的数组,制作路线的。
其余提供给handler块的参数是NSError对象,仿造下面的块处理方向回调:
[directions calculateDirectionsWithCompletionHandler:^(MKDirectionsResponse *response, NSError *error) { // Now handle the resultif (error) { NSLog(@"There was an error getting your directions"); return; } // So there wasn't an error - let's plot those routes_currentRoute = [response.routes firstObject]; [self plotRouteOnMap:_currentRoute]; }];
我们已经创建了一个实例方法在地图上绘制一个路线,下一节将会看到。
渲染polyline
我们已经收到路线的polyline,我们想把它体现到地图上。IOS7改变了地图渲染的方法,通过MKOverlayRenderer类。如果我们想做自定义形状或者非标准渲染技术,可以定义一个子类。但是,许多叠加渲染技术都是为标准用例使用的。我们想渲染一个polyline,可以使用对象MKPolylineRenderer。我们会稍后看一下何时何地创建渲染器,先看看上一节提到的plotRouteOnMap: 方法。
MKPolyline 对象代表多个段合成的一条线,遵循MKOverlay协议。也就是说,我们可以把它作为一个叠加层添加到MKMapView上,通过addOverlay:方法:
- (void)plotRouteOnMap:(MKRoute *)route { if(_routeOverlay) { [self.mapView removeOverlay:_routeOverlay]; } // Update the ivar_routeOverlay = route.polyline; // Add it to the map[self.mapView addOverlay:_routeOverlay]; }
这个方法带有MKRoute参数,通过属性mapView添加一个路线的polyline叠加层到MKMapView上。iavr变量_routeOverlay用来关联polyline。也就是说,当方法被调用时,可以移除一个已经存在的,然后用新的来代替它。
虽然我们已经添加了叠加层到地图View上,但是没有被画出来。这是因为地图不知道如何绘制这个叠加层对象,这也就是为什么要引入类MKOverlayRenderer 。当一个叠加层出现在地图view上,地图view会通过代理获取渲染器去绘制它。然后,当用户缩放和拖动地图时,渲染器就会根据不同的地图状态去绘制叠加层。
我们需要遵循MKMapViewDelegate协议,实现下面的方法来为地图view提供一个渲染器绘制polyline;
1 |
- (MKOverlayRenderer *)mapView:(MKMapView *)mapView rendererForOverlay:(id)overlay |
3 |
MKPolylineRenderer *renderer = [[MKPolylineRenderer alloc] initWithPolyline:overlay]; |
4 |
renderer.strokeColor = [UIColor redColor]; |
5 |
renderer.lineWidth = 4.0; |
我们已经获得一个有点简化了的案例,我们知道这里只会有一个叠加层,类型为MKPolyline,所以不需要任何代码去决定回调哪种渲染器。我们创建MKPolylineRenderer对象,它是MKOverlayRenderer的子类,目的是为了绘制polyline叠加层。我们设置了一些简单的属性(strokeColor 和 lineWidth),以便于可以看到叠加层,然后返回新的对象。
剩下的就是设置地图view的delegate属性,以便于当叠加层添加到地图时调用delegate里的方法:
5 |
self.mapView.delegate = self; |
搭建路由器
现在我们已经讨论了请求方向指南和获取响应的处理流程,但是没有给出太多关于本文应用的细节。虽然它没有进一步演示MapKit的细节,但是值得快速看一下它是如何构造出来的。
这个应用不是特别有用,因为它仅仅给出从当前位置到华盛顿特区的路线。应用中使用storyboard,并基于导航栏控制器。下面是app中包含的一些控制器:
1、 SCViewController。主视图。允许用户发送路线请求并当接收到响应时,把整个路线绘制到地图view上。它包含一个按钮(当接收到响应时显示)去查看详细路线,并推进下个视图控制器入栈。
2、 SCStepsViewController。这个是UITableViewController,每个cell展示路线的一段。点击cell将推进最后视图控制器入栈。
3、 SCIndividualStepViewController。展示特别的路段详情,包括地图,距离,路由服务器提供的一些介绍。
由于我们使用storyboards,每个视图控制器重写这个方法prepareForSegue:sender:,给下个视图提供展示需要的数据。例如,SCStepsViewController有一个route属性(MKRoute类型),就是在主视图跳转过来时设置的。
1 |
- ( void )prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender |
推荐阅读
-
2019独角兽企业重金招聘Python工程师标准#import*.h文件中导入以下两个框架*#import ...
[详细]
蜡笔小新 2023-09-16 18:32:13
-
在《Cocos2d-x学习笔记:基础概念解析与内存管理机制深入探讨》中,详细介绍了Cocos2d-x的基础概念,并深入分析了其内存管理机制。特别是针对Boost库引入的智能指针管理方法进行了详细的讲解,例如在处理鱼的运动过程中,可以通过编写自定义函数来动态计算角度变化,利用CallFunc回调机制实现高效的游戏逻辑控制。此外,文章还探讨了如何通过智能指针优化资源管理和避免内存泄漏,为开发者提供了实用的编程技巧和最佳实践。 ...
[详细]
蜡笔小新 2024-11-11 14:49:31
-
-
本文介绍了在 Java 编程中遇到的一个常见错误:对象无法转换为 long 类型,并提供了详细的解决方案。 ...
[详细]
蜡笔小新 2024-11-13 10:57:24
-
为了在Hadoop 2.7.2中实现对Snappy压缩和解压功能的原生支持,本文详细介绍了如何重新编译Hadoop源代码,并优化其Native编译过程。通过这一优化,可以显著提升数据处理的效率和性能。此外,还探讨了编译过程中可能遇到的问题及其解决方案,为用户提供了一套完整的操作指南。 ...
[详细]
蜡笔小新 2024-11-09 19:45:36
-
蜡笔小新 2024-11-02 16:38:16
-
添加#import头文件倒入mapkit.framework库mapkit.framework是属于ui,可以在故事版上添加mkmap ...
[详细]
蜡笔小新 2023-09-15 01:37:54
-
IOS笔记汇总为了方便开发者开发出强大的功能,苹果提供了各种各样的框架IOS属性IOS基础属性导入依赖propertyNSStringNSDictionaryNSAr ...
[详细]
蜡笔小新 2023-09-12 20:53:50
-
篇首语:本文由编程笔记#小编为大家整理,主要介绍了iOS核心笔记—CoreLocation框架-基础相关的知识,希望对你有一定的参考价值。1、 ...
[详细]
蜡笔小新 2023-08-28 19:27:18
-
1最近项目到了收尾阶段,所以开始了在真机部署调试,之前的模拟器环境下测试项目,一切都没问题,今天真机调试过程中, ...
[详细]
蜡笔小新 2023-08-24 16:51:39
-
本文介绍了如何通过类的约束来确保代码的一致性,以及如何使用异常处理和日志记录来提高代码的健壮性和可维护性。具体包括抛出异常、使用抽象类和方法,以及异常处理和日志记录的详细示例。 ...
[详细]
蜡笔小新 2024-11-14 06:38:46
-
为了满足项目需求,需要在多个场景中实现左滑删除功能,并且后续可能在列表项中增加其他功能。虽然网络上有很多左滑删除的示例,但大多数封装不够完善。因此,我们尝试自己封装一个更加灵活和通用的解决方案。 ...
[详细]
蜡笔小新 2024-11-13 17:43:59
-
HTTP协议是不加密传输数据的,也就是用户跟你的网站之间传递数据有可能在途中被截获,破解传递的真实内容,所以使用不加密的HTTP的网站是不 ...
[详细]
蜡笔小新 2024-11-13 14:02:50
-
本文详细介绍了在 CentOS 7 系统中配置 fstab 文件以实现开机自动挂载 NFS 共享目录的方法,并解决了常见的配置失败问题。 ...
[详细]
蜡笔小新 2024-11-13 12:05:24
-
1.java是面向对象的语言。设计模式:接口接口类是从java里衍生出来的,不是python原生支持的主要用于继承里多继承抽象类是python原生支持的主要用于继承里的单继承但是接 ...
[详细]
蜡笔小新 2024-11-12 06:43:20
-
如何在PHP中准确获取服务器IP地址? ...
[详细]
蜡笔小新 2024-11-10 15:17:16
-
可爱的你公馆_698
这个家伙很懒,什么也没留下!