iOS原生地图开发进阶——使用导航和附近兴趣点检索

转载 2016年08月31日 10:52:18

iOS中的mapKit框架对国际化的支持非常出色。在前些篇博客中,对这个地图框架的基础用法和标注与覆盖物的添加进行了详细的介绍,这篇博客将介绍两个更加实用的功能的开发:线路导航与兴趣点搜索。前几篇博客的链接如下:

地图基础用法详解:http://my.oschina.net/u/2340880/blog/415360

添加大头针与自定义标注:http://my.oschina.net/u/2340880/blog/415441

添加地图覆盖物:http://my.oschina.net/u/2340880/blog/415611

一、线路导航

1、从几个类的关系说起

(1)MKPlacemark

一个地点信息类,如下:

1
2
3
4
5
6
7
@interface MKPlacemark : CLPlacemark <MKAnnotation>
//初始化方法,通过给定一个经纬度和地点信息字典
- (instancetype)initWithCoordinate:(CLLocationCoordinate2D)coordinate
                 addressDictionary:(NSDictionary *)addressDictionary;
//国家编码
@property (nonatomic, readonly) NSString *countryCode;
@end

(2)MKMapItem

地点节点类,包含此节点的许多地点信息,如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
@interface MKMapItem : NSObject
//当前节点的地点信息对象
@property (nonatomic, readonly) MKPlacemark *placemark;
//是否是当前位置
@property (nonatomic, readonly) BOOL isCurrentLocation;
//节点名称
@property (nonatomic, copy) NSString *name;
//电话号码
@property (nonatomic, copy) NSString *phoneNumber;
//网址
@property (nonatomic, strong) NSURL *url;
//将当前位置创建为节点
+ (MKMapItem *)mapItemForCurrentLocation;
//由一个位置信息创建节点
- (instancetype)initWithPlacemark:(MKPlacemark *)placemark;
 
@end

(3)MKDirectionsRequest

导航请求类

1
2
3
4
5
6
7
8
9
@interface MKDirectionsRequest : NSObject
//起点节点
- (MKMapItem *)source NS_AVAILABLE(10_9, 6_0);
- (void)setSource:(MKMapItem *)source NS_AVAILABLE(10_9, 7_0);
//目的地节点
- (MKMapItem *)destination NS_AVAILABLE(10_9, 6_0);
- (void)setDestination:(MKMapItem *)destination NS_AVAILABLE(10_9, 7_0);
 
@end

这个类还有一些扩展的设置属性:

@property (nonatomicMKDirectionsTransportType transportType;

设置路线检索类型,枚举如下:

1
2
3
4
5
typedef NS_OPTIONS(NSUInteger, MKDirectionsTransportType) {
    MKDirectionsTransportTypeAutomobile     = 1 << 0,//适合驾车时导航
    MKDirectionsTransportTypeWalking        = 1 << 1,//适合步行时导航
    MKDirectionsTransportTypeAny            = 0x0FFFFFFF//任何情况
};

@property (nonatomicBOOL requestsAlternateRoutes;

设置是否搜索多条线路

@property (nonatomiccopyNSDate *departureDate;

设置出发日期

@property (nonatomiccopyNSDate *arrivalDate;

设置到达日期


(4)MKDirections

从apple服务器获取数据的连接类

1
2
3
4
5
6
7
8
9
10
11
12
@interface MKDirections : NSObject
//初始化方法
- (instancetype)initWithRequest:(MKDirectionsRequest *)request NS_DESIGNATED_INITIALIZER;
//开始计算线路信息
- (void)calculateDirectionsWithCompletionHandler:(MKDirectionsHandler)completionHandler;
//开始计算时间信息
- (void)calculateETAWithCompletionHandler:(MKETAHandler)completionHandler;
//取消
- (void)cancel;
//是否正在计算
@property (nonatomic, readonly, getter=isCalculating) BOOL calculating;
@end

(5)MKDirectionsResponse

线路信息结果类

1
2
3
4
5
@interface MKDirectionsResponse : NSObject
@property (nonatomic, readonly) MKMapItem *source;//起点
@property (nonatomic, readonly) MKMapItem *destination;//终点
@property (nonatomic, readonly) NSArray *routes; //线路规划数组
@end

(6)MKETResponse

时间信息结果类

1
2
3
4
5
6
@interface MKETAResponse : NSObject
@property (nonatomic, readonly) MKMapItem *source;//起点
@property (nonatomic, readonly) MKMapItem *destination;//终点
@property (nonatomic, readonly) NSTimeInterval expectedTravelTime;//耗时
 
@end

(7)MKRoute

线路信息类,导航的线路结果是这个类型的对象

1
2
3
4
5
6
7
8
9
10
11
12
13
@interface MKRoute : NSObject
 
@property (nonatomic, readonly) NSString *name; //线路名称
@property (nonatomic, readonly) NSArray *advisoryNotices; //注意事项
@property (nonatomic, readonly) CLLocationDistance distance; //距离
@property (nonatomic, readonly) NSTimeInterval expectedTravelTime;//耗时
@property (nonatomic, readonly) MKDirectionsTransportType transportType; //检索的类型
 
@property (nonatomic, readonly) MKPolyline *polyline; // 线路覆盖物
 
@property (nonatomic, readonly) NSArray *steps; // 线路详情数组
 
@end

(8)MKRouteStep

线路详情信息类,线路中每一步的信息都是这个类的对象

1
2
3
4
5
6
7
8
9
10
11
12
@interface MKRouteStep : NSObject
 
@property (nonatomic, readonly) NSString *instructions; // 节点信息
@property (nonatomic, readonly) NSString *notice; // 注意事项
 
@property (nonatomic, readonly) MKPolyline *polyline; //线路覆盖物
 
@property (nonatomic, readonly) CLLocationDistance distance; // 距离
 
@property (nonatomic, readonly) MKDirectionsTransportType transportType; // 导航类型
 
@end

看到上面如此多的类,你可能会觉得一头雾水,那么不用着急,类虽然繁杂,但他们之间的逻辑非常清晰,下面就通过一个例子来进行线路导航。

2、进行线路导航

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
- (void)viewDidLoad {
    [super viewDidLoad];
    //地图初始化设置
    mapView =[[MKMapView alloc]initWithFrame:self.view.frame];
    mapView.region=MKCoordinateRegionMake(CLLocationCoordinate2DMake(39.26, 116.3), MKCoordinateSpanMake(5, 5));
    mapView.mapType=MKMapTypeStandard;
    mapView.delegate=self;
    [self.view addSubview:mapView];
     
    //导航设置
    CLLocationCoordinate2D fromcoor=CLLocationCoordinate2DMake(39.26, 116.3);
    CLLocationCoordinate2D tocoor = CLLocationCoordinate2DMake(33.33, 113.33);
    //创建出发点和目的点信息
    MKPlacemark *fromPlace = [[MKPlacemark alloc] initWithCoordinate:fromcoor
                                                       addressDictionary:nil];
    MKPlacemark *toPlace = [[MKPlacemark alloc]initWithCoordinate:tocoor addressDictionary:nil];
    //创建出发节点和目的地节点
    MKMapItem * fromItem = [[MKMapItem alloc]initWithPlacemark:fromPlace];
    MKMapItem * toItem = [[MKMapItem alloc]initWithPlacemark:toPlace];
    //初始化导航搜索请求
    MKDirectionsRequest *request = [[MKDirectionsRequest alloc]init];
    request.source=fromItem;
    request.destination=toItem;
    request.requestsAlternateRoutes=YES;
    //初始化请求检索
    MKDirections *directions = [[MKDirections alloc]initWithRequest:request];
    //开始检索,结果会返回在block中
    [directions calculateDirectionsWithCompletionHandler:^(MKDirectionsResponse *response, NSError *error) {
        if (error) {
            NSLog(@"error:%@",error);
        }else{
            //提取导航线路结果中的一条线路
            MKRoute *route =response.routes[0];
            //将线路中的每一步详情提取出来
            NSArray * stepArray = [NSArray arrayWithArray:route.steps];
            //进行遍历
            for (int i=0; i<stepArray.count; i++) {
                //线路的详情节点
                MKRouteStep * step = stepArray[i];
                //在此节点处添加一个大头针
                MKPointAnnotation * point = [[MKPointAnnotation alloc]init];
                point.coordinate=step.polyline.coordinate;
                point.title=step.instructions;
                point.subtitle=step.notice;
                [mapView addAnnotation:point];
                //将此段线路添加到地图上
                [mapView addOverlay:step.polyline];
            }
        }
    }];  
}
//地图覆盖物的代理方法
-(MKOverlayRenderer *)mapView:(MKMapView *)mapView rendererForOverlay:(id<MKOverlay>)overlay{
    MKPolylineRenderer *renderer = [[MKPolylineRenderer alloc] initWithPolyline:overlay];
     
    renderer.strokeColor = [UIColor redColor];
     
    renderer.lineWidth = 4.0;
     
    return  renderer;
}
//标注的代理方法
-(MKAnnotationView *)mapView:(MKMapView *)mapView viewForAnnotation:(id<MKAnnotation>)annotation{
    MKPinAnnotationView * view= [[MKPinAnnotationView alloc]initWithAnnotation:annotation reuseIdentifier:@"anno"];
    view.canShowCallout=YES;
    return view;
}

效果如下:


二、附近兴趣点检索

兴趣点检索的逻辑和导航线路检索的逻辑相似,直接通过代码来演示:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
    //创建一个位置信息对象,第一个参数为经纬度,第二个为纬度检索范围,单位为米,第三个为经度检索范围,单位为米
    MKCoordinateRegion region = MKCoordinateRegionMakeWithDistance(tocoor, 5000, 5000);
    //初始化一个检索请求对象
    MKLocalSearchRequest * req = [[MKLocalSearchRequest alloc]init];
    //设置检索参数
    req.region=region;
    //兴趣点关键字
    req.naturalLanguageQuery=@"hotal";
    //初始化检索
    MKLocalSearch * ser = [[MKLocalSearch alloc]initWithRequest:req];
    //开始检索,结果返回在block中
    [ser startWithCompletionHandler:^(MKLocalSearchResponse *response, NSError *error) {
        //兴趣点节点数组
        NSArray * array = [NSArray arrayWithArray:response.mapItems];
        for (int i=0; i<array.count; i++) {
            MKMapItem * item=array[i];
            MKPointAnnotation * point = [[MKPointAnnotation alloc]init];
            point.title=item.name;
            point.subtitle=item.phoneNumber;
            point.coordinate=item.placemark.coordinate;
            [mapView addAnnotation:point];
        }
    }];

效果如下:

相关文章推荐

iOS 百度地图的检索(正向、反向、周边)

正向反向检索:功能描述: (1)我们在项目中往往会需要根据经纬度在地图中定位一个地点,在哪里插个大头针展示下; (2)当经纬度没有的时候,可以根据具体的地址信息去获取经纬度在地图上插个大头针展示下...

iOS原生地图开发进阶——使用导航和附近兴趣点检索

转载自:http://www.sxt.cn/info-6041-u-7372.html iOS原生地图开发进阶——使用导航和附近兴趣点检索 iOS中的mapKit框架对国际化的支持非常出色...

iOS开发:地图关键词搜索 MKLocal​Search使用

MKCoordinateRegion region = _mapView.region; MKLocalSearchRequest *localSearchRequest = [[MKLoca...

iOS原生地图热点搜索和基本使用汇总

1. 创建之前的准备工作   1.1  在Info.plist文件中添加下面两句话   NSLocationAlwaysUsageDescription —> 确定定位吗?亲   //请...

iOS 调起地图App进行导航(百度,高德,系统自带高德)URL API方式

在开发中需要用到地图。有两种方法,一种是导入地图 (百度,高德)地图 SDK,一种是使用 URL  API。 但是需要用到一些高级功能的时候,如果使用 SDK 开发,就是一件折磨人的事情。 SDK 还...

iOS - 高德地图API简单开发(一)

最近看了看高德地图的开发平台以及它们的API,奇怪的是一直没找到语音API,这个问题放到以后再说吧。 对于高德地图的开发平台,它们提供的说明文档还算健全,而且由于API是国人开发的,所以所有接口都有中...

ios原生地图开发篇

开发环境:Xcode6.4 模拟器  : IOS8.4 OS X : 10.0.4 小编博客链接: http://www.goofyy.com 首先例子是小编写的一个定位获取经纬度,然后在地图上面显示...

ios-关于使用高德地图api心得(1)

1.获取当前屏幕中心点的经纬度MACoordinateRegion region; CLLocationCoordinate2D centerCoordinate = mapView.region.c...

MapKit之地理编码和反向地理编码(CLGeocoder、CLPlacemark)

地理编码和反向地理编码    CLGeocoder类主要提供的服务是在具体经纬度信息和用户位置信息之间的转换。    CLGeocoder将根据请求的所有信息决定返回什么类型的信息。比如:...

android 百度地图SDK 获取手机附近POI兴趣点列表 (过时)

文章内容已经过时~大家去百度官方api学习。 http://developer.baidu.com/map/ http://lbsyun.baidu.com/sdk/download ...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:iOS原生地图开发进阶——使用导航和附近兴趣点检索
举报原因:
原因补充:

(最多只允许输入30个字)